diff --git a/discord/app_commands/commands.py b/discord/app_commands/commands.py index d1098a923..59b9c16af 100644 --- a/discord/app_commands/commands.py +++ b/discord/app_commands/commands.py @@ -88,9 +88,10 @@ else: T = TypeVar('T') GroupT = TypeVar('GroupT', bound='Union[Group, Cog]') Coro = Coroutine[Any, Any, T] +UnboundError = Callable[['Interaction', AppCommandError], Coro[Any]] Error = Union[ Callable[[GroupT, 'Interaction', AppCommandError], Coro[Any]], - Callable[['Interaction', AppCommandError], Coro[Any]], + UnboundError, ] Check = Callable[['Interaction'], Union[bool, Coro[bool]]] @@ -727,6 +728,7 @@ class ContextMenu: self._annotation = annotation self.module: Optional[str] = callback.__module__ self._guild_ids = guild_ids + self.on_error: Optional[UnboundError] = None self.checks: List[Check] = getattr(callback, '__discord_app_commands_checks__', []) @property @@ -746,6 +748,7 @@ class ContextMenu: self._annotation = annotation self.module = callback.__module__ self._guild_ids = None + self.on_error = None self.checks = getattr(callback, '__discord_app_commands_checks__', []) return self @@ -774,6 +777,32 @@ class ContextMenu: except Exception as e: raise CommandInvokeError(self, e) from e + def error(self, coro: UnboundError) -> UnboundError: + """A decorator that registers a coroutine as a local error handler. + + The local error handler is called whenever an exception is raised in the body + of the command or during handling of the command. The error handler must take + 2 parameters, the interaction and the error. + + The error passed will be derived from :exc:`AppCommandError`. + + Parameters + ----------- + coro: :ref:`coroutine ` + The coroutine to register as the local error handler. + + Raises + ------- + TypeError + The coroutine passed is not actually a coroutine. + """ + + if not inspect.iscoroutinefunction(coro): + raise TypeError('The error handler must be a coroutine.') + + self.on_error = coro + return coro + def add_check(self, func: Check, /) -> None: """Adds a check to the command. diff --git a/discord/app_commands/tree.py b/discord/app_commands/tree.py index 98e52bf9c..fb728d0c1 100644 --- a/discord/app_commands/tree.py +++ b/discord/app_commands/tree.py @@ -963,6 +963,8 @@ class CommandTree(Generic[ClientT]): try: await ctx_menu._invoke(interaction, value) except AppCommandError as e: + if ctx_menu.on_error is not None: + await ctx_menu.on_error(interaction, e) await self.on_error(interaction, ctx_menu, e) async def call(self, interaction: Interaction) -> None: