Source code for discord.ext.bridge.core

"""
The MIT License (MIT)

Copyright (c) 2015-2021 Rapptz
Copyright (c) 2021-present Pycord Development

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

import inspect
from collections.abc import Iterator
from typing import TYPE_CHECKING, Any, Callable

import discord.commands.options
from discord import (
    ApplicationCommand,
    Attachment,
    Option,
    Permissions,
    SlashCommand,
    SlashCommandGroup,
    SlashCommandOptionType,
)

from ...utils import MISSING, find, get
from ..commands import BadArgument
from ..commands import Bot as ExtBot
from ..commands import (
    Command,
    Context,
    Converter,
    Group,
    GuildChannelConverter,
    RoleConverter,
    UserConverter,
)
from ..commands.converter import _convert_to_bool, run_converters

if TYPE_CHECKING:
    from .context import BridgeApplicationContext, BridgeExtContext


__all__ = (
    "BridgeCommand",
    "BridgeCommandGroup",
    "bridge_command",
    "bridge_group",
    "BridgeExtCommand",
    "BridgeSlashCommand",
    "BridgeExtGroup",
    "BridgeSlashGroup",
    "BridgeOption",
    "map_to",
    "guild_only",
    "has_permissions",
    "is_nsfw",
)


[docs]class BridgeSlashCommand(SlashCommand): """A subclass of :class:`.SlashCommand` that is used for bridge commands.""" def __init__(self, func, **kwargs): self.brief = kwargs.pop("brief", None) super().__init__(func, **kwargs) async def dispatch_error( self, ctx: BridgeApplicationContext, error: Exception ) -> None: await super().dispatch_error(ctx, error) ctx.bot.dispatch("bridge_command_error", ctx, error)
[docs]class BridgeExtCommand(Command): """A subclass of :class:`.ext.commands.Command` that is used for bridge commands.""" def __init__(self, func, **kwargs): super().__init__(func, **kwargs) async def dispatch_error(self, ctx: BridgeExtContext, error: Exception) -> None: await super().dispatch_error(ctx, error) ctx.bot.dispatch("bridge_command_error", ctx, error) async def transform(self, ctx: Context, param: inspect.Parameter) -> Any: if param.annotation is Attachment: # skip the parameter checks for bridge attachments return await run_converters(ctx, AttachmentConverter, None, param) else: return await super().transform(ctx, param)
[docs]class BridgeSlashGroup(SlashCommandGroup): """A subclass of :class:`.SlashCommandGroup` that is used for bridge commands.""" __slots__ = ("module",) def __init__(self, callback, *args, **kwargs): if perms := getattr(callback, "__default_member_permissions__", None): kwargs["default_member_permissions"] = perms super().__init__(*args, **kwargs) self.callback = callback self.__original_kwargs__["callback"] = callback self.__command = None async def _invoke(self, ctx: BridgeApplicationContext) -> None: if not (options := ctx.interaction.data.get("options")): if not self.__command: self.__command = BridgeSlashCommand(self.callback) ctx.command = self.__command return await ctx.command.invoke(ctx) option = options[0] resolved = ctx.interaction.data.get("resolved", None) command = find(lambda x: x.name == option["name"], self.subcommands) option["resolved"] = resolved ctx.interaction.data = option await command.invoke(ctx)
[docs]class BridgeExtGroup(BridgeExtCommand, Group): """A subclass of :class:`.ext.commands.Group` that is used for bridge commands."""
[docs]class BridgeCommand: """Compatibility class between prefixed-based commands and slash commands. Parameters ---------- callback: Callable[[:class:`.BridgeContext`, ...], Awaitable[Any]] The callback to invoke when the command is executed. The first argument will be a :class:`BridgeContext`, and any additional arguments will be passed to the callback. This callback must be a coroutine. parent: Optional[:class:`.BridgeCommandGroup`]: Parent of the BridgeCommand. kwargs: Optional[Dict[:class:`str`, Any]] Keyword arguments that are directly passed to the respective command constructors. (:class:`.SlashCommand` and :class:`.ext.commands.Command`) Attributes ---------- slash_variant: :class:`.BridgeSlashCommand` The slash command version of this bridge command. ext_variant: :class:`.BridgeExtCommand` The prefix-based version of this bridge command. """ __special_attrs__ = ["slash_variant", "ext_variant", "parent"] def __init__(self, callback, **kwargs): self.parent = kwargs.pop("parent", None) self.slash_variant: BridgeSlashCommand = kwargs.pop( "slash_variant", None ) or BridgeSlashCommand(callback, **kwargs) self.ext_variant: BridgeExtCommand = kwargs.pop( "ext_variant", None ) or BridgeExtCommand(callback, **kwargs) @property def name_localizations(self) -> dict[str, str] | None: """Returns name_localizations from :attr:`slash_variant` You can edit/set name_localizations directly with .. code-block:: python3 bridge_command.name_localizations["en-UK"] = ... # or any other locale # or bridge_command.name_localizations = {"en-UK": ..., "fr-FR": ...} """ return self.slash_variant.name_localizations @name_localizations.setter def name_localizations(self, value): self.slash_variant.name_localizations = value @property def description_localizations(self) -> dict[str, str] | None: """Returns description_localizations from :attr:`slash_variant` You can edit/set description_localizations directly with .. code-block:: python3 bridge_command.description_localizations["en-UK"] = ... # or any other locale # or bridge_command.description_localizations = {"en-UK": ..., "fr-FR": ...} """ return self.slash_variant.description_localizations @description_localizations.setter def description_localizations(self, value): self.slash_variant.description_localizations = value def __getattribute__(self, name): try: # first, look for the attribute on the bridge command return super().__getattribute__(name) except AttributeError as e: # if it doesn't exist, check this list, if the name of # the parameter is here if name is self.__special_attrs__: raise e # looks up the result in the variants. # slash cmd prioritized result = getattr(self.slash_variant, name, MISSING) try: if result is MISSING: return getattr(self.ext_variant, name) return result except AttributeError: raise AttributeError( f"'{self.__class__.__name__}' object has no attribute '{name}'" ) def __setattr__(self, name, value) -> None: if name not in self.__special_attrs__: setattr(self.slash_variant, name, value) setattr(self.ext_variant, name, value) return super().__setattr__(name, value)
[docs] def add_to(self, bot: ExtBot) -> None: """Adds the command to a bot. This method is inherited by :class:`.BridgeCommandGroup`. Parameters ---------- bot: Union[:class:`.Bot`, :class:`.AutoShardedBot`] The bot to add the command to. """ bot.add_application_command(self.slash_variant) bot.add_command(self.ext_variant)
async def invoke( self, ctx: BridgeExtContext | BridgeApplicationContext, /, *args, **kwargs ): if ctx.is_app: return await self.slash_variant.invoke(ctx) return await self.ext_variant.invoke(ctx)
[docs] def error(self, coro): """A decorator that registers a coroutine as a local error handler. This error handler is limited to the command it is defined to. However, higher scope handlers (per-cog and global) are still invoked afterwards as a catch-all. This handler also functions as the handler for both the prefixed and slash versions of the command. This error handler takes two parameters, a :class:`.BridgeContext` and a :class:`~discord.DiscordException`. Parameters ---------- coro: :ref:`coroutine <coroutine>` The coroutine to register as the local error handler. Raises ------ TypeError The coroutine passed is not actually a coroutine. """ self.slash_variant.error(coro) self.ext_variant.on_error = coro return coro
[docs] def before_invoke(self, coro): """A decorator that registers a coroutine as a pre-invoke hook. This hook is called directly before the command is called, making it useful for any sort of set up required. This hook is called for both the prefixed and slash versions of the command. This pre-invoke hook takes a sole parameter, a :class:`.BridgeContext`. Parameters ---------- coro: :ref:`coroutine <coroutine>` The coroutine to register as the pre-invoke hook. Raises ------ TypeError The coroutine passed is not actually a coroutine. """ self.slash_variant.before_invoke(coro) self.ext_variant._before_invoke = coro return coro
[docs] def after_invoke(self, coro): """A decorator that registers a coroutine as a post-invoke hook. This hook is called directly after the command is called, making it useful for any sort of clean up required. This hook is called for both the prefixed and slash versions of the command. This post-invoke hook takes a sole parameter, a :class:`.BridgeContext`. Parameters ---------- coro: :ref:`coroutine <coroutine>` The coroutine to register as the post-invoke hook. Raises ------ TypeError The coroutine passed is not actually a coroutine. """ self.slash_variant.after_invoke(coro) self.ext_variant._after_invoke = coro return coro
[docs]class BridgeCommandGroup(BridgeCommand): """Compatibility class between prefixed-based commands and slash commands. Parameters ---------- callback: Callable[[:class:`.BridgeContext`, ...], Awaitable[Any]] The callback to invoke when the command is executed. The first argument will be a :class:`BridgeContext`, and any additional arguments will be passed to the callback. This callback must be a coroutine. kwargs: Optional[Dict[:class:`str`, Any]] Keyword arguments that are directly passed to the respective command constructors. (:class:`.SlashCommand` and :class:`.ext.commands.Command`) Attributes ---------- slash_variant: :class:`.SlashCommandGroup` The slash command version of this command group. ext_variant: :class:`.ext.commands.Group` The prefix-based version of this command group. subcommands: List[:class:`.BridgeCommand`] List of bridge commands in this group mapped: Optional[:class:`.SlashCommand`] If :func:`map_to` is used, the mapped slash command. """ __special_attrs__ = [ "slash_variant", "ext_variant", "parent", "subcommands", "mapped", ] ext_variant: BridgeExtGroup slash_variant: BridgeSlashGroup def __init__(self, callback, *args, **kwargs): ext_var = BridgeExtGroup(callback, *args, **kwargs) kwargs.update({"name": ext_var.name}) super().__init__( callback, ext_variant=ext_var, slash_variant=BridgeSlashGroup(callback, *args, **kwargs), parent=kwargs.pop("parent", None), ) self.subcommands: list[BridgeCommand] = [] self.mapped: SlashCommand | None = None if map_to := getattr(callback, "__custom_map_to__", None): kwargs.update(map_to) self.mapped = self.slash_variant.command(**kwargs)(callback)
[docs] def walk_commands(self) -> Iterator[BridgeCommand]: """An iterator that recursively walks through all the bridge group's subcommands. Yields ------ :class:`.BridgeCommand` A bridge command of this bridge group. """ yield from self.subcommands
[docs] def command(self, *args, **kwargs): """A decorator to register a function as a subcommand. Parameters ---------- kwargs: Optional[Dict[:class:`str`, Any]] Keyword arguments that are directly passed to the respective command constructors. (:class:`.SlashCommand` and :class:`.ext.commands.Command`) """ def wrap(callback): slash = self.slash_variant.command( *args, **kwargs, cls=BridgeSlashCommand, )(callback) ext = self.ext_variant.command( *args, **kwargs, cls=BridgeExtCommand, )(callback) command = BridgeCommand( callback, parent=self, slash_variant=slash, ext_variant=ext ) self.subcommands.append(command) return command return wrap
def bridge_command(**kwargs): """A decorator that is used to wrap a function as a bridge command. Parameters ---------- kwargs: Optional[Dict[:class:`str`, Any]] Keyword arguments that are directly passed to the respective command constructors. (:class:`.SlashCommand` and :class:`.ext.commands.Command`) """ def decorator(callback): return BridgeCommand(callback, **kwargs) return decorator def bridge_group(**kwargs): """A decorator that is used to wrap a function as a bridge command group. Parameters ---------- kwargs: Optional[Dict[:class:`str`, Any]] Keyword arguments that are directly passed to the respective command constructors (:class:`.SlashCommandGroup` and :class:`.ext.commands.Group`). """ def decorator(callback): return BridgeCommandGroup(callback, **kwargs) return decorator def map_to(name, description=None): """To be used with bridge command groups, map the main command to a slash subcommand. Parameters ---------- name: :class:`str` The new name of the mapped command. description: Optional[:class:`str`] The new description of the mapped command. Example ------- .. code-block:: python3 @bot.bridge_group() @bridge.map_to("show") async def config(ctx: BridgeContext): ... @config.command() async def toggle(ctx: BridgeContext): ... Prefixed commands will not be affected, but slash commands will appear as: .. code-block:: /config show /config toggle """ def decorator(callback): callback.__custom_map_to__ = {"name": name, "description": description} return callback return decorator def guild_only(): """Intended to work with :class:`.ApplicationCommand` and :class:`BridgeCommand`, adds a :func:`~ext.commands.check` that locks the command to only run in guilds, and also registers the command as guild only client-side (on discord). Basically a utility function that wraps both :func:`discord.ext.commands.guild_only` and :func:`discord.commands.guild_only`. """ def predicate(func: Callable | ApplicationCommand): if isinstance(func, ApplicationCommand): func.guild_only = True else: func.__guild_only__ = True from ..commands import guild_only return guild_only()(func) return predicate def is_nsfw(): """Intended to work with :class:`.ApplicationCommand` and :class:`BridgeCommand`, adds a :func:`~ext.commands.check` that locks the command to only run in nsfw contexts, and also registers the command as nsfw client-side (on discord). Basically a utility function that wraps both :func:`discord.ext.commands.is_nsfw` and :func:`discord.commands.is_nsfw`. .. warning:: In DMs, the prefixed-based command will always run as the user's privacy settings cannot be checked directly. """ def predicate(func: Callable | ApplicationCommand): if isinstance(func, ApplicationCommand): func.nsfw = True else: func.__nsfw__ = True from ..commands import is_nsfw return is_nsfw()(func) return predicate def has_permissions(**perms: bool): r"""Intended to work with :class:`.SlashCommand` and :class:`BridgeCommand`, adds a :func:`~ext.commands.check` that locks the command to be run by people with certain permissions inside guilds, and also registers the command as locked behind said permissions. Basically a utility function that wraps both :func:`discord.ext.commands.has_permissions` and :func:`discord.commands.default_permissions`. Parameters ---------- \*\*perms: Dict[:class:`str`, :class:`bool`] An argument list of permissions to check for. """ def predicate(func: Callable | ApplicationCommand): from ..commands import has_permissions func = has_permissions(**perms)(func) _perms = Permissions(**perms) if isinstance(func, ApplicationCommand): func.default_member_permissions = _perms else: func.__default_member_permissions__ = _perms return func return predicate class MentionableConverter(Converter): """A converter that can convert a mention to a user or a role.""" async def convert(self, ctx, argument): try: return await RoleConverter().convert(ctx, argument) except BadArgument: return await UserConverter().convert(ctx, argument) class AttachmentConverter(Converter): async def convert(self, ctx: Context, arg: str): try: attach = ctx.message.attachments[0] except IndexError: raise BadArgument("At least 1 attachment is needed") else: return attach class BooleanConverter(Converter): async def convert(self, ctx, arg: bool): return _convert_to_bool(str(arg)) BRIDGE_CONVERTER_MAPPING = { SlashCommandOptionType.string: str, SlashCommandOptionType.integer: int, SlashCommandOptionType.boolean: BooleanConverter, SlashCommandOptionType.user: UserConverter, SlashCommandOptionType.channel: GuildChannelConverter, SlashCommandOptionType.role: RoleConverter, SlashCommandOptionType.mentionable: MentionableConverter, SlashCommandOptionType.number: float, SlashCommandOptionType.attachment: AttachmentConverter, }
[docs]class BridgeOption(Option, Converter): """A subclass of :class:`discord.Option` which represents a selectable slash command option and a prefixed command argument for bridge commands. """
[docs] async def convert(self, ctx, argument: str) -> Any: try: if self.converter is not None: converted = await self.converter.convert(ctx, argument) else: converter = BRIDGE_CONVERTER_MAPPING[self.input_type] if issubclass(converter, Converter): converted = await converter().convert(ctx, argument) # type: ignore # protocol class else: converted = converter(argument) if self.choices: choices_names: list[str | int | float] = [ choice.name for choice in self.choices ] if converted in choices_names and ( choice := get(self.choices, name=converted) ): converted = choice.value else: choices = [choice.value for choice in self.choices] if converted not in choices: raise ValueError( f"{argument} is not a valid choice. Valid choices:" f" {list(set(choices_names + choices))}" ) return converted except ValueError as exc: raise BadArgument() from exc