| 1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
 | import logging
import math
import random
import sys
import traceback
from discord import Colour, Embed, Message
from discord.ext import commands
from bot.constants import NEGATIVE_REPLIES
from bot.decorators import InChannelCheckFailure
log = logging.getLogger(__name__)
class CommandErrorHandler(commands.Cog):
    """A error handler for the PythonDiscord server."""
    def __init__(self, bot: commands.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."
            )
    @commands.Cog.listener()
    async def on_command_error(self, ctx: commands.Context, error: commands.CommandError) -> None:
        """Activates when a command opens an error."""
        if hasattr(ctx.command, 'on_error'):
            return logging.debug(
                "A command error occured but the command had it's own error handler."
            )
        error = getattr(error, 'original', error)
        if isinstance(error, InChannelCheckFailure):
            logging.debug(
                f"{ctx.author} the command '{ctx.command}', but they did not have "
                f"permissions to run commands in the channel {ctx.channel}!"
            )
            embed = Embed(colour=Colour.red())
            embed.title = random.choice(NEGATIVE_REPLIES)
            embed.description = str(error)
            return await ctx.send(embed=embed)
        if isinstance(error, commands.CommandNotFound):
            return logging.debug(
                f"{ctx.author} called '{ctx.message.content}' but no command was found."
            )
        if isinstance(error, commands.UserInputError):
            logging.debug(
                f"{ctx.author} called the command '{ctx.command}' but entered invalid input!"
            )
            self.revert_cooldown_counter(ctx.command, ctx.message)
            return await ctx.send(
                ":no_entry: The command you specified failed to run. "
                "This is because the arguments you provided were invalid."
            )
        if isinstance(error, commands.CommandOnCooldown):
            logging.debug(
                f"{ctx.author} called the command '{ctx.command}' but they were on cooldown!"
            )
            remaining_minutes, remaining_seconds = divmod(error.retry_after, 60)
            return await ctx.send(
                "This command is on cooldown, please retry in "
                f"{int(remaining_minutes)} minutes {math.ceil(remaining_seconds)} seconds."
            )
        if isinstance(error, commands.DisabledCommand):
            logging.debug(
                f"{ctx.author} called the command '{ctx.command}' but the command was disabled!"
            )
            return await ctx.send(":no_entry: This command has been disabled.")
        if isinstance(error, commands.NoPrivateMessage):
            logging.debug(
                f"{ctx.author} called the command '{ctx.command}' "
                "in a private message however the command was guild only!"
            )
            return await ctx.author.send(":no_entry: This command can only be used in the server.")
        if isinstance(error, commands.BadArgument):
            self.revert_cooldown_counter(ctx.command, ctx.message)
            logging.debug(
                f"{ctx.author} called the command '{ctx.command}' but entered a bad argument!"
            )
            return await ctx.send("The argument you provided was invalid.")
        if isinstance(error, commands.CheckFailure):
            logging.debug(f"{ctx.author} called the command '{ctx.command}' but the checks failed!")
            return await ctx.send(":no_entry: You are not authorized to use this command.")
        print(f"Ignoring exception in command {ctx.command}:", file=sys.stderr)
        logging.warning(
            f"{ctx.author} called the command '{ctx.command}' "
            "however the command failed to run with the error:"
            f"-------------\n{error}"
        )
        traceback.print_exception(type(error), error, error.__traceback__, file=sys.stderr)
def setup(bot: commands.Bot) -> None:
    """Error handler Cog load."""
    bot.add_cog(CommandErrorHandler(bot))
    log.info("CommandErrorHandler cog loaded")
 |