diff options
Diffstat (limited to 'bot/exts/evergreen/error_handler.py')
| -rw-r--r-- | bot/exts/evergreen/error_handler.py | 182 | 
1 files changed, 0 insertions, 182 deletions
| diff --git a/bot/exts/evergreen/error_handler.py b/bot/exts/evergreen/error_handler.py deleted file mode 100644 index fd2123e7..00000000 --- a/bot/exts/evergreen/error_handler.py +++ /dev/null @@ -1,182 +0,0 @@ -import difflib -import logging -import math -import random -from collections.abc import Iterable -from typing import Union - -from discord import Embed, Message -from discord.ext import commands -from sentry_sdk import push_scope - -from bot.bot import Bot -from bot.constants import Channels, Colours, ERROR_REPLIES, NEGATIVE_REPLIES, RedirectOutput -from bot.utils.decorators import InChannelCheckFailure, InMonthCheckFailure -from bot.utils.exceptions import APIError, UserNotPlayingError - -log = logging.getLogger(__name__) - - -QUESTION_MARK_ICON = "https://cdn.discordapp.com/emojis/512367613339369475.png" - - -class CommandErrorHandler(commands.Cog): -    """A error handler for the PythonDiscord server.""" - -    def __init__(self, bot: Bot): -        self.bot = bot - -    @staticmethod -    def revert_cooldown_counter(command: commands.Command, message: Message) -> None: -        """Undoes the last cooldown counter for user-error cases.""" -        if command._buckets.valid: -            bucket = command._buckets.get_bucket(message) -            bucket._tokens = min(bucket.rate, bucket._tokens + 1) -            logging.debug("Cooldown counter reverted as the command was not used correctly.") - -    @staticmethod -    def error_embed(message: str, title: Union[Iterable, str] = ERROR_REPLIES) -> Embed: -        """Build a basic embed with red colour and either a random error title or a title provided.""" -        embed = Embed(colour=Colours.soft_red) -        if isinstance(title, str): -            embed.title = title -        else: -            embed.title = random.choice(title) -        embed.description = message -        return embed - -    @commands.Cog.listener() -    async def on_command_error(self, ctx: commands.Context, error: commands.CommandError) -> None: -        """Activates when a command raises an error.""" -        if getattr(error, "handled", False): -            logging.debug(f"Command {ctx.command} had its error already handled locally; ignoring.") -            return - -        parent_command = "" -        if subctx := getattr(ctx, "subcontext", None): -            parent_command = f"{ctx.command} " -            ctx = subctx - -        error = getattr(error, "original", error) -        logging.debug( -            f"Error Encountered: {type(error).__name__} - {str(error)}, " -            f"Command: {ctx.command}, " -            f"Author: {ctx.author}, " -            f"Channel: {ctx.channel}" -        ) - -        if isinstance(error, commands.CommandNotFound): -            await self.send_command_suggestion(ctx, ctx.invoked_with) -            return - -        if isinstance(error, (InChannelCheckFailure, InMonthCheckFailure)): -            await ctx.send(embed=self.error_embed(str(error), NEGATIVE_REPLIES), delete_after=7.5) -            return - -        if isinstance(error, commands.UserInputError): -            self.revert_cooldown_counter(ctx.command, ctx.message) -            usage = f"```\n{ctx.prefix}{parent_command}{ctx.command} {ctx.command.signature}\n```" -            embed = self.error_embed( -                f"Your input was invalid: {error}\n\nUsage:{usage}" -            ) -            await ctx.send(embed=embed) -            return - -        if isinstance(error, commands.CommandOnCooldown): -            mins, secs = divmod(math.ceil(error.retry_after), 60) -            embed = self.error_embed( -                f"This command is on cooldown:\nPlease retry in {mins} minutes {secs} seconds.", -                NEGATIVE_REPLIES -            ) -            await ctx.send(embed=embed, delete_after=7.5) -            return - -        if isinstance(error, commands.DisabledCommand): -            await ctx.send(embed=self.error_embed("This command has been disabled.", NEGATIVE_REPLIES)) -            return - -        if isinstance(error, commands.NoPrivateMessage): -            await ctx.send( -                embed=self.error_embed( -                    f"This command can only be used in the server. Go to <#{Channels.community_bot_commands}> instead!", -                    NEGATIVE_REPLIES -                ) -            ) -            return - -        if isinstance(error, commands.BadArgument): -            self.revert_cooldown_counter(ctx.command, ctx.message) -            embed = self.error_embed( -                "The argument you provided was invalid: " -                f"{error}\n\nUsage:\n```\n{ctx.prefix}{parent_command}{ctx.command} {ctx.command.signature}\n```" -            ) -            await ctx.send(embed=embed) -            return - -        if isinstance(error, commands.CheckFailure): -            await ctx.send(embed=self.error_embed("You are not authorized to use this command.", NEGATIVE_REPLIES)) -            return - -        if isinstance(error, UserNotPlayingError): -            await ctx.send("Game not found.") -            return - -        if isinstance(error, APIError): -            await ctx.send( -                embed=self.error_embed( -                    f"There was an error when communicating with the {error.api}", -                    NEGATIVE_REPLIES -                ) -            ) -            return - -        with push_scope() as scope: -            scope.user = { -                "id": ctx.author.id, -                "username": str(ctx.author) -            } - -            scope.set_tag("command", ctx.command.qualified_name) -            scope.set_tag("message_id", ctx.message.id) -            scope.set_tag("channel_id", ctx.channel.id) - -            scope.set_extra("full_message", ctx.message.content) - -            if ctx.guild is not None: -                scope.set_extra("jump_to", ctx.message.jump_url) - -            log.exception(f"Unhandled command error: {str(error)}", exc_info=error) - -    async def send_command_suggestion(self, ctx: commands.Context, command_name: str) -> None: -        """Sends user similar commands if any can be found.""" -        raw_commands = [] -        for cmd in self.bot.walk_commands(): -            if not cmd.hidden: -                raw_commands += (cmd.name, *cmd.aliases) -        if similar_command_data := difflib.get_close_matches(command_name, raw_commands, 1): -            similar_command_name = similar_command_data[0] -            similar_command = self.bot.get_command(similar_command_name) - -            if not similar_command: -                return - -            log_msg = "Cancelling attempt to suggest a command due to failed checks." -            try: -                if not await similar_command.can_run(ctx): -                    log.debug(log_msg) -                    return -            except commands.errors.CommandError as cmd_error: -                log.debug(log_msg) -                await self.on_command_error(ctx, cmd_error) -                return - -            misspelled_content = ctx.message.content -            e = Embed() -            e.set_author(name="Did you mean:", icon_url=QUESTION_MARK_ICON) -            e.description = misspelled_content.replace(command_name, similar_command_name, 1) -            await ctx.send(embed=e, delete_after=RedirectOutput.delete_delay) - - -def setup(bot: Bot) -> None: -    """Load the ErrorHandler cog.""" -    bot.add_cog(CommandErrorHandler(bot)) | 
