diff options
57 files changed, 1204 insertions, 842 deletions
| diff --git a/Dockerfile b/Dockerfile index c285898dc..4d8592590 100644 --- a/Dockerfile +++ b/Dockerfile @@ -1,4 +1,4 @@ -FROM python:3.9.5-slim +FROM python:3.9-slim  # Set pip to have no saved cache  ENV PIP_NO_CACHE_DIR=false \ diff --git a/bot/constants.py b/bot/constants.py index 500803f33..12b5c02e5 100644 --- a/bot/constants.py +++ b/bot/constants.py @@ -568,7 +568,7 @@ class Metabase(metaclass=YAMLGetter):      username: Optional[str]      password: Optional[str] -    url: str +    base_url: str      max_session_age: int diff --git a/bot/converters.py b/bot/converters.py index 595809517..37eb91c7f 100644 --- a/bot/converters.py +++ b/bot/converters.py @@ -2,7 +2,6 @@ import logging  import re  import typing as t  from datetime import datetime -from functools import partial  from ssl import CertificateError  import dateutil.parser @@ -439,29 +438,6 @@ class HushDurationConverter(Converter):          return duration -def proxy_user(user_id: str) -> discord.Object: -    """ -    Create a proxy user object from the given id. - -    Used when a Member or User object cannot be resolved. -    """ -    log.trace(f"Attempting to create a proxy user for the user id {user_id}.") - -    try: -        user_id = int(user_id) -    except ValueError: -        log.debug(f"Failed to create proxy user {user_id}: could not convert to int.") -        raise BadArgument(f"User ID `{user_id}` is invalid - could not convert to an integer.") - -    user = discord.Object(user_id) -    user.mention = user.id -    user.display_name = f"<@{user.id}>" -    user.avatar_url_as = lambda static_format: None -    user.bot = False - -    return user - -  class UserMentionOrID(UserConverter):      """      Converts to a `discord.User`, but only if a mention or userID is provided. @@ -480,64 +456,6 @@ class UserMentionOrID(UserConverter):              raise BadArgument(f"`{argument}` is not a User mention or a User ID.") -class FetchedUser(UserConverter): -    """ -    Converts to a `discord.User` or, if it fails, a `discord.Object`. - -    Unlike the default `UserConverter`, which only does lookups via the global user cache, this -    converter attempts to fetch the user via an API call to Discord when the using the cache is -    unsuccessful. - -    If the fetch also fails and the error doesn't imply the user doesn't exist, then a -    `discord.Object` is returned via the `user_proxy` converter. - -    The lookup strategy is as follows (in order): - -    1. Lookup by ID. -    2. Lookup by mention. -    3. Lookup by name#discrim -    4. Lookup by name -    5. Lookup via API -    6. Create a proxy user with discord.Object -    """ - -    async def convert(self, ctx: Context, arg: str) -> t.Union[discord.User, discord.Object]: -        """Convert the `arg` to a `discord.User` or `discord.Object`.""" -        try: -            return await super().convert(ctx, arg) -        except BadArgument: -            pass - -        try: -            user_id = int(arg) -            log.trace(f"Fetching user {user_id}...") -            return await ctx.bot.fetch_user(user_id) -        except ValueError: -            log.debug(f"Failed to fetch user {arg}: could not convert to int.") -            raise BadArgument(f"The provided argument can't be turned into integer: `{arg}`") -        except discord.HTTPException as e: -            # If the Discord error isn't `Unknown user`, return a proxy instead -            if e.code != 10013: -                log.info(f"Failed to fetch user, returning a proxy instead: status {e.status}") -                return proxy_user(arg) - -            log.debug(f"Failed to fetch user {arg}: user does not exist.") -            raise BadArgument(f"User `{arg}` does not exist") - - -def _snowflake_from_regex(pattern: t.Pattern, arg: str) -> int: -    """ -    Extract the snowflake from `arg` using a regex `pattern` and return it as an int. - -    The snowflake is expected to be within the first capture group in `pattern`. -    """ -    match = pattern.match(arg) -    if not match: -        raise BadArgument(f"Mention {str!r} is invalid.") - -    return int(match.group(1)) - -  class Infraction(Converter):      """      Attempts to convert a given infraction ID into an infraction. @@ -568,5 +486,4 @@ class Infraction(Converter):  Expiry = t.Union[Duration, ISODateTime] -FetchedMember = t.Union[discord.Member, FetchedUser] -UserMention = partial(_snowflake_from_regex, RE_USER_MENTION) +MemberOrUser = t.Union[discord.Member, discord.User] diff --git a/bot/errors.py b/bot/errors.py index 46efb6d4f..08396ec3e 100644 --- a/bot/errors.py +++ b/bot/errors.py @@ -1,6 +1,6 @@ -from typing import Hashable, Union +from typing import Hashable -from discord import Member, User +from bot.converters import MemberOrUser  class LockedResourceError(RuntimeError): @@ -30,7 +30,8 @@ class InvalidInfractedUserError(Exception):          `user` -- User or Member which is invalid      """ -    def __init__(self, user: Union[Member, User], reason: str = "User infracted is a bot."): +    def __init__(self, user: MemberOrUser, reason: str = "User infracted is a bot."): +          self.user = user          self.reason = reason @@ -41,3 +42,17 @@ class BrandingMisconfiguration(RuntimeError):      """Raised by the Branding cog when a misconfigured event is encountered."""      pass + + +class NonExistentRoleError(ValueError): +    """ +    Raised by the Information Cog when encountering a Role that does not exist. + +    Attributes: +        `role_id` -- the ID of the role that does not exist +    """ + +    def __init__(self, role_id: int): +        super().__init__(f"Could not fetch data for role {role_id}") + +        self.role_id = role_id diff --git a/bot/exts/backend/branding/_cog.py b/bot/exts/backend/branding/_cog.py index 47c379a34..0ba146635 100644 --- a/bot/exts/backend/branding/_cog.py +++ b/bot/exts/backend/branding/_cog.py @@ -50,7 +50,7 @@ def make_embed(title: str, description: str, *, success: bool) -> discord.Embed:      For both `title` and `description`, empty string are valid values ~ fields will be empty.      """      colour = Colours.soft_green if success else Colours.soft_red -    return discord.Embed(title=title[:256], description=description[:2048], colour=colour) +    return discord.Embed(title=title[:256], description=description[:4096], colour=colour)  def extract_event_duration(event: Event) -> str: @@ -293,8 +293,8 @@ class Branding(commands.Cog):          else:              content = "Python Discord is entering a new event!" if is_notification else None -            embed = discord.Embed(description=description[:2048], colour=discord.Colour.blurple()) -            embed.set_footer(text=duration[:2048]) +            embed = discord.Embed(description=description[:4096], colour=discord.Colour.blurple()) +            embed.set_footer(text=duration[:4096])          await channel.send(content=content, embed=embed) diff --git a/bot/exts/events/__init__.py b/bot/exts/events/__init__.py new file mode 100644 index 000000000..e69de29bb --- /dev/null +++ b/bot/exts/events/__init__.py diff --git a/bot/exts/events/code_jams/__init__.py b/bot/exts/events/code_jams/__init__.py new file mode 100644 index 000000000..16e81e365 --- /dev/null +++ b/bot/exts/events/code_jams/__init__.py @@ -0,0 +1,8 @@ +from bot.bot import Bot + + +def setup(bot: Bot) -> None: +    """Load the CodeJams cog.""" +    from bot.exts.events.code_jams._cog import CodeJams + +    bot.add_cog(CodeJams(bot)) diff --git a/bot/exts/events/code_jams/_channels.py b/bot/exts/events/code_jams/_channels.py new file mode 100644 index 000000000..34ff0ad41 --- /dev/null +++ b/bot/exts/events/code_jams/_channels.py @@ -0,0 +1,113 @@ +import logging +import typing as t + +import discord + +from bot.constants import Categories, Channels, Roles + +log = logging.getLogger(__name__) + +MAX_CHANNELS = 50 +CATEGORY_NAME = "Code Jam" + + +async def _get_category(guild: discord.Guild) -> discord.CategoryChannel: +    """ +    Return a code jam category. + +    If all categories are full or none exist, create a new category. +    """ +    for category in guild.categories: +        if category.name == CATEGORY_NAME and len(category.channels) < MAX_CHANNELS: +            return category + +    return await _create_category(guild) + + +async def _create_category(guild: discord.Guild) -> discord.CategoryChannel: +    """Create a new code jam category and return it.""" +    log.info("Creating a new code jam category.") + +    category_overwrites = { +        guild.default_role: discord.PermissionOverwrite(read_messages=False), +        guild.me: discord.PermissionOverwrite(read_messages=True) +    } + +    category = await guild.create_category_channel( +        CATEGORY_NAME, +        overwrites=category_overwrites, +        reason="It's code jam time!" +    ) + +    await _send_status_update( +        guild, f"Created a new category with the ID {category.id} for this Code Jam's team channels." +    ) + +    return category + + +def _get_overwrites( +        members: list[tuple[discord.Member, bool]], +        guild: discord.Guild, +) -> dict[t.Union[discord.Member, discord.Role], discord.PermissionOverwrite]: +    """Get code jam team channels permission overwrites.""" +    team_channel_overwrites = { +        guild.default_role: discord.PermissionOverwrite(read_messages=False), +        guild.get_role(Roles.code_jam_event_team): discord.PermissionOverwrite(read_messages=True) +    } + +    for member, _ in members: +        team_channel_overwrites[member] = discord.PermissionOverwrite( +            read_messages=True +        ) + +    return team_channel_overwrites + + +async def create_team_channel( +        guild: discord.Guild, +        team_name: str, +        members: list[tuple[discord.Member, bool]], +        team_leaders: discord.Role +) -> None: +    """Create the team's text channel.""" +    await _add_team_leader_roles(members, team_leaders) + +    # Get permission overwrites and category +    team_channel_overwrites = _get_overwrites(members, guild) +    code_jam_category = await _get_category(guild) + +    # Create a text channel for the team +    await code_jam_category.create_text_channel( +        team_name, +        overwrites=team_channel_overwrites, +    ) + + +async def create_team_leader_channel(guild: discord.Guild, team_leaders: discord.Role) -> None: +    """Create the Team Leader Chat channel for the Code Jam team leaders.""" +    category: discord.CategoryChannel = guild.get_channel(Categories.summer_code_jam) + +    team_leaders_chat = await category.create_text_channel( +        name="team-leaders-chat", +        overwrites={ +            guild.default_role: discord.PermissionOverwrite(read_messages=False), +            team_leaders: discord.PermissionOverwrite(read_messages=True) +        } +    ) + +    await _send_status_update(guild, f"Created {team_leaders_chat.mention} in the {category} category.") + + +async def _send_status_update(guild: discord.Guild, message: str) -> None: +    """Inform the events lead with a status update when the command is ran.""" +    channel: discord.TextChannel = guild.get_channel(Channels.code_jam_planning) + +    await channel.send(f"<@&{Roles.events_lead}>\n\n{message}") + + +async def _add_team_leader_roles(members: list[tuple[discord.Member, bool]], team_leaders: discord.Role) -> None: +    """Assign the team leader role to the team leaders.""" +    for member, is_leader in members: +        if is_leader: +            await member.add_roles(team_leaders) diff --git a/bot/exts/events/code_jams/_cog.py b/bot/exts/events/code_jams/_cog.py new file mode 100644 index 000000000..e099f7dfa --- /dev/null +++ b/bot/exts/events/code_jams/_cog.py @@ -0,0 +1,235 @@ +import asyncio +import csv +import logging +import typing as t +from collections import defaultdict + +import discord +from discord import Colour, Embed, Guild, Member +from discord.ext import commands + +from bot.bot import Bot +from bot.constants import Emojis, Roles +from bot.exts.events.code_jams import _channels +from bot.utils.services import send_to_paste_service + +log = logging.getLogger(__name__) + +TEAM_LEADERS_COLOUR = 0x11806a +DELETION_REACTION = "\U0001f4a5" + + +class CodeJams(commands.Cog): +    """Manages the code-jam related parts of our server.""" + +    def __init__(self, bot: Bot): +        self.bot = bot + +    @commands.group(aliases=("cj", "jam")) +    @commands.has_any_role(Roles.admins) +    async def codejam(self, ctx: commands.Context) -> None: +        """A Group of commands for managing Code Jams.""" +        if ctx.invoked_subcommand is None: +            await ctx.send_help(ctx.command) + +    @codejam.command() +    async def create(self, ctx: commands.Context, csv_file: t.Optional[str] = None) -> None: +        """ +        Create code-jam teams from a CSV file or a link to one, specifying the team names, leaders and members. + +        The CSV file must have 3 columns: 'Team Name', 'Team Member Discord ID', and 'Team Leader'. + +        This will create the text channels for the teams, and give the team leaders their roles. +        """ +        async with ctx.typing(): +            if csv_file: +                async with self.bot.http_session.get(csv_file) as response: +                    if response.status != 200: +                        await ctx.send(f"Got a bad response from the URL: {response.status}") +                        return + +                    csv_file = await response.text() + +            elif ctx.message.attachments: +                csv_file = (await ctx.message.attachments[0].read()).decode("utf8") +            else: +                raise commands.BadArgument("You must include either a CSV file or a link to one.") + +            teams = defaultdict(list) +            reader = csv.DictReader(csv_file.splitlines()) + +            for row in reader: +                member = ctx.guild.get_member(int(row["Team Member Discord ID"])) + +                if member is None: +                    log.trace(f"Got an invalid member ID: {row['Team Member Discord ID']}") +                    continue + +                teams[row["Team Name"]].append((member, row["Team Leader"].upper() == "Y")) + +            team_leaders = await ctx.guild.create_role(name="Code Jam Team Leaders", colour=TEAM_LEADERS_COLOUR) + +            for team_name, members in teams.items(): +                await _channels.create_team_channel(ctx.guild, team_name, members, team_leaders) + +            await _channels.create_team_leader_channel(ctx.guild, team_leaders) +            await ctx.send(f"{Emojis.check_mark} Created Code Jam with {len(teams)} teams.") + +    @codejam.command() +    @commands.has_any_role(Roles.admins) +    async def end(self, ctx: commands.Context) -> None: +        """ +        Delete all code jam channels. + +        A confirmation message is displayed with the categories and channels to be deleted.. Pressing the added reaction +        deletes those channels. +        """ +        def predicate_deletion_emoji_reaction(reaction: discord.Reaction, user: discord.User) -> bool: +            """Return True if the reaction :boom: was added by the context message author on this message.""" +            return ( +                reaction.message.id == message.id +                and user.id == ctx.author.id +                and str(reaction) == DELETION_REACTION +            ) + +        # A copy of the list of channels is stored. This is to make sure that we delete precisely the channels displayed +        # in the confirmation message. +        categories = self.jam_categories(ctx.guild) +        category_channels = {category: category.channels.copy() for category in categories} + +        confirmation_message = await self._build_confirmation_message(category_channels) +        message = await ctx.send(confirmation_message) +        await message.add_reaction(DELETION_REACTION) +        try: +            await self.bot.wait_for( +                'reaction_add', +                check=predicate_deletion_emoji_reaction, +                timeout=10 +            ) + +        except asyncio.TimeoutError: +            await message.clear_reaction(DELETION_REACTION) +            await ctx.send("Command timed out.", reference=message) +            return + +        else: +            await message.clear_reaction(DELETION_REACTION) +            for category, channels in category_channels.items(): +                for channel in channels: +                    await channel.delete(reason="Code jam ended.") +                await category.delete(reason="Code jam ended.") + +            await message.add_reaction(Emojis.check_mark) + +    @staticmethod +    async def _build_confirmation_message( +        categories: dict[discord.CategoryChannel, list[discord.abc.GuildChannel]] +    ) -> str: +        """Sends details of the channels to be deleted to the pasting service, and formats the confirmation message.""" +        def channel_repr(channel: discord.abc.GuildChannel) -> str: +            """Formats the channel name and ID and a readable format.""" +            return f"{channel.name} ({channel.id})" + +        def format_category_info(category: discord.CategoryChannel, channels: list[discord.abc.GuildChannel]) -> str: +            """Displays the category and the channels within it in a readable format.""" +            return f"{channel_repr(category)}:\n" + "\n".join("  - " + channel_repr(channel) for channel in channels) + +        deletion_details = "\n\n".join( +            format_category_info(category, channels) for category, channels in categories.items() +        ) + +        url = await send_to_paste_service(deletion_details) +        if url is None: +            url = "**Unable to send deletion details to the pasting service.**" + +        return f"Are you sure you want to delete all code jam channels?\n\nThe channels to be deleted: {url}" + +    @codejam.command() +    @commands.has_any_role(Roles.admins, Roles.code_jam_event_team) +    async def info(self, ctx: commands.Context, member: Member) -> None: +        """ +        Send an info embed about the member with the team they're in. + +        The team is found by searching the permissions of the team channels. +        """ +        channel = self.team_channel(ctx.guild, member) +        if not channel: +            await ctx.send(":x: I can't find the team channel for this member.") +            return + +        embed = Embed( +            title=str(member), +            colour=Colour.blurple() +        ) +        embed.add_field(name="Team", value=self.team_name(channel), inline=True) + +        await ctx.send(embed=embed) + +    @codejam.command() +    @commands.has_any_role(Roles.admins) +    async def move(self, ctx: commands.Context, member: Member, new_team_name: str) -> None: +        """Move participant from one team to another by changing the user's permissions for the relevant channels.""" +        old_team_channel = self.team_channel(ctx.guild, member) +        if not old_team_channel: +            await ctx.send(":x: I can't find the team channel for this member.") +            return + +        if old_team_channel.name == new_team_name or self.team_name(old_team_channel) == new_team_name: +            await ctx.send(f"`{member}` is already in `{new_team_name}`.") +            return + +        new_team_channel = self.team_channel(ctx.guild, new_team_name) +        if not new_team_channel: +            await ctx.send(f":x: I can't find a team channel named `{new_team_name}`.") +            return + +        await old_team_channel.set_permissions(member, overwrite=None, reason=f"Participant moved to {new_team_name}") +        await new_team_channel.set_permissions( +            member, +            overwrite=discord.PermissionOverwrite(read_messages=True), +            reason=f"Participant moved from {old_team_channel.name}" +        ) + +        await ctx.send( +            f"Participant moved from `{self.team_name(old_team_channel)}` to `{self.team_name(new_team_channel)}`." +        ) + +    @codejam.command() +    @commands.has_any_role(Roles.admins) +    async def remove(self, ctx: commands.Context, member: Member) -> None: +        """Remove the participant from their team. Does not remove the participants or leader roles.""" +        channel = self.team_channel(ctx.guild, member) +        if not channel: +            await ctx.send(":x: I can't find the team channel for this member.") +            return + +        await channel.set_permissions( +            member, +            overwrite=None, +            reason=f"Participant removed from the team  {self.team_name(channel)}." +        ) +        await ctx.send(f"Removed the participant from `{self.team_name(channel)}`.") + +    @staticmethod +    def jam_categories(guild: Guild) -> list[discord.CategoryChannel]: +        """Get all the code jam team categories.""" +        return [category for category in guild.categories if category.name == _channels.CATEGORY_NAME] + +    @staticmethod +    def team_channel(guild: Guild, criterion: t.Union[str, Member]) -> t.Optional[discord.TextChannel]: +        """Get a team channel through either a participant or the team name.""" +        for category in CodeJams.jam_categories(guild): +            for channel in category.channels: +                if isinstance(channel, discord.TextChannel): +                    if ( +                        # If it's a string. +                        criterion == channel.name or criterion == CodeJams.team_name(channel) +                        # If it's a member. +                        or criterion in channel.overwrites +                    ): +                        return channel + +    @staticmethod +    def team_name(channel: discord.TextChannel) -> str: +        """Retrieves the team name from the given channel.""" +        return channel.name.replace("-", " ").title() diff --git a/bot/exts/filters/antimalware.py b/bot/exts/filters/antimalware.py index 4c4836c88..0eedeb0fb 100644 --- a/bot/exts/filters/antimalware.py +++ b/bot/exts/filters/antimalware.py @@ -7,7 +7,7 @@ from discord.ext.commands import Cog  from bot.bot import Bot  from bot.constants import Channels, Filter, URLs -from bot.exts.utils.jams import CATEGORY_NAME as JAM_CATEGORY_NAME +from bot.exts.events.code_jams._channels import CATEGORY_NAME as JAM_CATEGORY_NAME  log = logging.getLogger(__name__) diff --git a/bot/exts/filters/antispam.py b/bot/exts/filters/antispam.py index 48c3aa5a6..226da2790 100644 --- a/bot/exts/filters/antispam.py +++ b/bot/exts/filters/antispam.py @@ -17,8 +17,8 @@ from bot.constants import (      Guild as GuildConfig, Icons,  )  from bot.converters import Duration +from bot.exts.events.code_jams._channels import CATEGORY_NAME as JAM_CATEGORY_NAME  from bot.exts.moderation.modlog import ModLog -from bot.exts.utils.jams import CATEGORY_NAME as JAM_CATEGORY_NAME  from bot.utils import lock, scheduling  from bot.utils.messages import format_user, send_attachments @@ -85,7 +85,7 @@ class DeletionContext:              mod_alert_message += "Message:\n"              [message] = self.messages.values()              content = message.clean_content -            remaining_chars = 2040 - len(mod_alert_message) +            remaining_chars = 4080 - len(mod_alert_message)              if len(content) > remaining_chars:                  content = content[:remaining_chars] + "..." diff --git a/bot/exts/filters/filtering.py b/bot/exts/filters/filtering.py index 16aaf11cf..10cc7885d 100644 --- a/bot/exts/filters/filtering.py +++ b/bot/exts/filters/filtering.py @@ -19,8 +19,8 @@ from bot.constants import (      Channels, Colours, Filter,      Guild, Icons, URLs  ) +from bot.exts.events.code_jams._channels import CATEGORY_NAME as JAM_CATEGORY_NAME  from bot.exts.moderation.modlog import ModLog -from bot.exts.utils.jams import CATEGORY_NAME as JAM_CATEGORY_NAME  from bot.utils.messages import format_user  from bot.utils.regex import INVITE_RE  from bot.utils.scheduling import Scheduler diff --git a/bot/exts/filters/webhook_remover.py b/bot/exts/filters/webhook_remover.py index f11fc8912..25e267426 100644 --- a/bot/exts/filters/webhook_remover.py +++ b/bot/exts/filters/webhook_remover.py @@ -9,12 +9,15 @@ from bot.constants import Channels, Colours, Event, Icons  from bot.exts.moderation.modlog import ModLog  from bot.utils.messages import format_user -WEBHOOK_URL_RE = re.compile(r"((?:https?://)?discord(?:app)?\.com/api/webhooks/\d+/)\S+/?", re.IGNORECASE) +WEBHOOK_URL_RE = re.compile( +    r"((?:https?:\/\/)?(?:ptb\.|canary\.)?discord(?:app)?\.com\/api\/webhooks\/\d+\/)\S+\/?", +    re.IGNORECASE +)  ALERT_MESSAGE_TEMPLATE = (      "{user}, looks like you posted a Discord webhook URL. Therefore, your " -    "message has been removed. Your webhook may have been **compromised** so " -    "please re-create the webhook **immediately**. If you believe this was a " +    "message has been removed, and your webhook has been deleted. " +    "You can re-create it if you wish to. If you believe this was a "      "mistake, please let us know."  ) @@ -32,7 +35,7 @@ class WebhookRemover(Cog):          """Get current instance of `ModLog`."""          return self.bot.get_cog("ModLog") -    async def delete_and_respond(self, msg: Message, redacted_url: str) -> None: +    async def delete_and_respond(self, msg: Message, redacted_url: str, *, webhook_deleted: bool) -> None:          """Delete `msg` and send a warning that it contained the Discord webhook `redacted_url`."""          # Don't log this, due internal delete, not by user. Will make different entry.          self.mod_log.ignore(Event.message_delete, msg.id) @@ -44,9 +47,12 @@ class WebhookRemover(Cog):              return          await msg.channel.send(ALERT_MESSAGE_TEMPLATE.format(user=msg.author.mention)) - +        if webhook_deleted: +            delete_state = "The webhook was successfully deleted." +        else: +            delete_state = "There was an error when deleting the webhook, it might have already been removed."          message = ( -            f"{format_user(msg.author)} posted a Discord webhook URL to {msg.channel.mention}. " +            f"{format_user(msg.author)} posted a Discord webhook URL to {msg.channel.mention}. {delete_state} "              f"Webhook URL was `{redacted_url}`"          )          log.debug(message) @@ -72,7 +78,10 @@ class WebhookRemover(Cog):          matches = WEBHOOK_URL_RE.search(msg.content)          if matches: -            await self.delete_and_respond(msg, matches[1] + "xxx") +            async with self.bot.http_session.delete(matches[0]) as resp: +                # The Discord API Returns a 204 NO CONTENT response on success. +                deleted_successfully = resp.status == 204 +            await self.delete_and_respond(msg, matches[1] + "xxx", webhook_deleted=deleted_successfully)      @Cog.listener()      async def on_message_edit(self, before: Message, after: Message) -> None: diff --git a/bot/exts/fun/duck_pond.py b/bot/exts/fun/duck_pond.py index c78b9c141..7f7e4585c 100644 --- a/bot/exts/fun/duck_pond.py +++ b/bot/exts/fun/duck_pond.py @@ -3,11 +3,12 @@ import logging  from typing import Union  import discord -from discord import Color, Embed, Member, Message, RawReactionActionEvent, TextChannel, User, errors +from discord import Color, Embed, Message, RawReactionActionEvent, TextChannel, errors  from discord.ext.commands import Cog, Context, command  from bot import constants  from bot.bot import Bot +from bot.converters import MemberOrUser  from bot.utils.checks import has_any_role  from bot.utils.messages import count_unique_users_reaction, send_attachments  from bot.utils.webhooks import send_webhook @@ -36,7 +37,7 @@ class DuckPond(Cog):              log.exception(f"Failed to fetch webhook with id `{self.webhook_id}`")      @staticmethod -    def is_staff(member: Union[User, Member]) -> bool: +    def is_staff(member: MemberOrUser) -> bool:          """Check if a specific member or user is staff."""          if hasattr(member, "roles"):              for role in member.roles: @@ -171,8 +172,14 @@ class DuckPond(Cog):          if not self.is_helper_viewable(channel):              return -        message = await channel.fetch_message(payload.message_id) +        try: +            message = await channel.fetch_message(payload.message_id) +        except discord.NotFound: +            return  # Message was deleted. +          member = discord.utils.get(message.guild.members, id=payload.user_id) +        if not member: +            return  # Member left or wasn't in the cache.          # Was the message sent by a human staff member?          if not self.is_staff(message.author) or message.author.bot: diff --git a/bot/exts/help_channels/_cog.py b/bot/exts/help_channels/_cog.py index 35658d117..cfc9cf477 100644 --- a/bot/exts/help_channels/_cog.py +++ b/bot/exts/help_channels/_cog.py @@ -267,6 +267,8 @@ class HelpChannels(commands.Cog):              for channel in channels[:abs(missing)]:                  await self.unclaim_channel(channel, closed_on=_channel.ClosingReason.CLEANUP) +        self.available_help_channels = set(_channel.get_category_channels(self.available_category)) +          # Getting channels that need to be included in the dynamic message.          await self.update_available_help_channels()          log.trace("Dynamic available help message updated.") @@ -387,7 +389,12 @@ class HelpChannels(commands.Cog):          )          log.trace(f"Sending dormant message for #{channel} ({channel.id}).") -        embed = discord.Embed(description=_message.DORMANT_MSG) +        embed = discord.Embed( +            description=_message.DORMANT_MSG.format( +                dormant=self.dormant_category.name, +                available=self.available_category.name, +            ) +        )          await channel.send(embed=embed)          log.trace(f"Pushing #{channel} ({channel.id}) into the channel queue.") @@ -511,11 +518,6 @@ class HelpChannels(commands.Cog):      async def update_available_help_channels(self) -> None:          """Updates the dynamic message within #how-to-get-help for available help channels.""" -        if not self.available_help_channels: -            self.available_help_channels = set( -                c for c in self.available_category.channels if not _channel.is_excluded_channel(c) -            ) -          available_channels = AVAILABLE_HELP_CHANNELS.format(              available=", ".join(                  c.mention for c in sorted(self.available_help_channels, key=attrgetter("position")) diff --git a/bot/exts/help_channels/_message.py b/bot/exts/help_channels/_message.py index befacd263..077b20b47 100644 --- a/bot/exts/help_channels/_message.py +++ b/bot/exts/help_channels/_message.py @@ -30,12 +30,12 @@ AVAILABLE_TITLE = "Available help channel"  AVAILABLE_FOOTER = "Closes after a period of inactivity, or when you send !close."  DORMANT_MSG = f""" -This help channel has been marked as **dormant**, and has been moved into the **Help: Dormant** \ +This help channel has been marked as **dormant**, and has been moved into the **{{dormant}}** \  category at the bottom of the channel list. It is no longer possible to send messages in this \  channel until it becomes available again.  If your question wasn't answered yet, you can claim a new help channel from the \ -**Help: Available** category by simply asking your question again. Consider rephrasing the \ +**{{available}}** category by simply asking your question again. Consider rephrasing the \  question to maximize your chance of getting a good answer. If you're not sure how, have a look \  through our guide for **[asking a good question]({ASKING_GUIDE_URL})**.  """ diff --git a/bot/exts/info/code_snippets.py b/bot/exts/info/code_snippets.py index 24a9ae28a..4a90a0668 100644 --- a/bot/exts/info/code_snippets.py +++ b/bot/exts/info/code_snippets.py @@ -4,8 +4,8 @@ import textwrap  from typing import Any  from urllib.parse import quote_plus +import discord  from aiohttp import ClientResponseError -from discord import Message  from discord.ext.commands import Cog  from bot.bot import Bot @@ -45,6 +45,17 @@ class CodeSnippets(Cog):      Matches each message against a regex and prints the contents of all matched snippets.      """ +    def __init__(self, bot: Bot): +        """Initializes the cog's bot.""" +        self.bot = bot + +        self.pattern_handlers = [ +            (GITHUB_RE, self._fetch_github_snippet), +            (GITHUB_GIST_RE, self._fetch_github_gist_snippet), +            (GITLAB_RE, self._fetch_gitlab_snippet), +            (BITBUCKET_RE, self._fetch_bitbucket_snippet) +        ] +      async def _fetch_response(self, url: str, response_format: str, **kwargs) -> Any:          """Makes http requests using aiohttp."""          async with self.bot.http_session.get(url, raise_for_status=True, **kwargs) as response: @@ -208,56 +219,56 @@ class CodeSnippets(Cog):          # Returns an empty codeblock if the snippet is empty          return f'{ret}``` ```' -    def __init__(self, bot: Bot): -        """Initializes the cog's bot.""" -        self.bot = bot +    async def _parse_snippets(self, content: str) -> str: +        """Parse message content and return a string with a code block for each URL found.""" +        all_snippets = [] + +        for pattern, handler in self.pattern_handlers: +            for match in pattern.finditer(content): +                try: +                    snippet = await handler(**match.groupdict()) +                    all_snippets.append((match.start(), snippet)) +                except ClientResponseError as error: +                    error_message = error.message  # noqa: B306 +                    log.log( +                        logging.DEBUG if error.status == 404 else logging.ERROR, +                        f'Failed to fetch code snippet from {match[0]!r}: {error.status} ' +                        f'{error_message} for GET {error.request_info.real_url.human_repr()}' +                    ) -        self.pattern_handlers = [ -            (GITHUB_RE, self._fetch_github_snippet), -            (GITHUB_GIST_RE, self._fetch_github_gist_snippet), -            (GITLAB_RE, self._fetch_gitlab_snippet), -            (BITBUCKET_RE, self._fetch_bitbucket_snippet) -        ] +        # Sorts the list of snippets by their match index and joins them into a single message +        return '\n'.join(map(lambda x: x[1], sorted(all_snippets)))      @Cog.listener() -    async def on_message(self, message: Message) -> None: +    async def on_message(self, message: discord.Message) -> None:          """Checks if the message has a snippet link, removes the embed, then sends the snippet contents.""" -        if not message.author.bot: -            all_snippets = [] - -            for pattern, handler in self.pattern_handlers: -                for match in pattern.finditer(message.content): -                    try: -                        snippet = await handler(**match.groupdict()) -                        all_snippets.append((match.start(), snippet)) -                    except ClientResponseError as error: -                        error_message = error.message  # noqa: B306 -                        log.log( -                            logging.DEBUG if error.status == 404 else logging.ERROR, -                            f'Failed to fetch code snippet from {match[0]!r}: {error.status} ' -                            f'{error_message} for GET {error.request_info.real_url.human_repr()}' -                        ) - -            # Sorts the list of snippets by their match index and joins them into a single message -            message_to_send = '\n'.join(map(lambda x: x[1], sorted(all_snippets))) - -            if 0 < len(message_to_send) <= 2000 and message_to_send.count('\n') <= 15: +        if message.author.bot: +            return + +        message_to_send = await self._parse_snippets(message.content) +        destination = message.channel + +        if 0 < len(message_to_send) <= 2000 and message_to_send.count('\n') <= 15: +            try:                  await message.edit(suppress=True) -                if len(message_to_send) > 1000 and message.channel.id != Channels.bot_commands: -                    # Redirects to #bot-commands if the snippet contents are too long -                    await self.bot.wait_until_guild_available() -                    await message.channel.send(('The snippet you tried to send was too long. Please ' -                                                f'see <#{Channels.bot_commands}> for the full snippet.')) -                    bot_commands_channel = self.bot.get_channel(Channels.bot_commands) -                    await wait_for_deletion( -                        await bot_commands_channel.send(message_to_send), -                        (message.author.id,) -                    ) -                else: -                    await wait_for_deletion( -                        await message.channel.send(message_to_send), -                        (message.author.id,) -                    ) +            except discord.NotFound: +                # Don't send snippets if the original message was deleted. +                return + +            if len(message_to_send) > 1000 and message.channel.id != Channels.bot_commands: +                # Redirects to #bot-commands if the snippet contents are too long +                await self.bot.wait_until_guild_available() +                destination = self.bot.get_channel(Channels.bot_commands) + +                await message.channel.send( +                    'The snippet you tried to send was too long. ' +                    f'Please see {destination.mention} for the full snippet.' +                ) + +            await wait_for_deletion( +                await destination.send(message_to_send), +                (message.author.id,) +            )  def setup(bot: Bot) -> None: diff --git a/bot/exts/info/codeblock/_cog.py b/bot/exts/info/codeblock/_cog.py index 9094d9d15..9a0705d2b 100644 --- a/bot/exts/info/codeblock/_cog.py +++ b/bot/exts/info/codeblock/_cog.py @@ -177,10 +177,13 @@ class CodeBlockCog(Cog, name="Code Block"):          if not bot_message:              return -        if not instructions: -            log.info("User's incorrect code block has been fixed. Removing instructions message.") -            await bot_message.delete() -            del self.codeblock_message_ids[payload.message_id] -        else: -            log.info("Message edited but still has invalid code blocks; editing the instructions.") -            await bot_message.edit(embed=self.create_embed(instructions)) +        try: +            if not instructions: +                log.info("User's incorrect code block was fixed. Removing instructions message.") +                await bot_message.delete() +                del self.codeblock_message_ids[payload.message_id] +            else: +                log.info("Message edited but still has invalid code blocks; editing instructions.") +                await bot_message.edit(embed=self.create_embed(instructions)) +        except discord.NotFound: +            log.debug("Could not find instructions message; it was probably deleted.") diff --git a/bot/exts/info/doc/_cog.py b/bot/exts/info/doc/_cog.py index c54a3ee1c..fb9b2584a 100644 --- a/bot/exts/info/doc/_cog.py +++ b/bot/exts/info/doc/_cog.py @@ -341,10 +341,13 @@ class DocCog(commands.Cog):              if doc_embed is None:                  error_message = await send_denial(ctx, "No documentation found for the requested symbol.")                  await wait_for_deletion(error_message, (ctx.author.id,), timeout=NOT_FOUND_DELETE_DELAY) -                with suppress(discord.NotFound): -                    await ctx.message.delete() -                with suppress(discord.NotFound): -                    await error_message.delete() + +                # Make sure that we won't cause a ghost-ping by deleting the message +                if not (ctx.message.mentions or ctx.message.role_mentions): +                    with suppress(discord.NotFound): +                        await ctx.message.delete() +                        await error_message.delete() +              else:                  msg = await ctx.send(embed=doc_embed)                  await wait_for_deletion(msg, (ctx.author.id,)) diff --git a/bot/exts/info/doc/_parsing.py b/bot/exts/info/doc/_parsing.py index bf840b96f..1a0d42c47 100644 --- a/bot/exts/info/doc/_parsing.py +++ b/bot/exts/info/doc/_parsing.py @@ -34,7 +34,7 @@ _EMBED_CODE_BLOCK_LINE_LENGTH = 61  # _MAX_SIGNATURE_AMOUNT code block wrapped lines with py syntax highlight  _MAX_SIGNATURES_LENGTH = (_EMBED_CODE_BLOCK_LINE_LENGTH + 8) * MAX_SIGNATURE_AMOUNT  # Maximum embed description length - signatures on top -_MAX_DESCRIPTION_LENGTH = 2048 - _MAX_SIGNATURES_LENGTH +_MAX_DESCRIPTION_LENGTH = 4096 - _MAX_SIGNATURES_LENGTH  _TRUNCATE_STRIP_CHARACTERS = "!?:;." + string.whitespace  BracketPair = namedtuple("BracketPair", ["opening_bracket", "closing_bracket"]) diff --git a/bot/exts/info/information.py b/bot/exts/info/information.py index fc3c2c61e..8bef6a8cd 100644 --- a/bot/exts/info/information.py +++ b/bot/exts/info/information.py @@ -3,21 +3,23 @@ import logging  import pprint  import textwrap  from collections import defaultdict -from typing import Any, DefaultDict, Dict, Mapping, Optional, Tuple, Union +from typing import Any, DefaultDict, Mapping, Optional, Tuple, Union  import rapidfuzz  from discord import AllowedMentions, Colour, Embed, Guild, Message, Role  from discord.ext.commands import BucketType, Cog, Context, Paginator, command, group, has_any_role +from discord.utils import escape_markdown  from bot import constants  from bot.api import ResponseCodeError  from bot.bot import Bot -from bot.converters import FetchedMember +from bot.converters import MemberOrUser  from bot.decorators import in_whitelist +from bot.errors import NonExistentRoleError  from bot.pagination import LinePaginator  from bot.utils.channel import is_mod_channel, is_staff_channel  from bot.utils.checks import cooldown_with_role_bypass, has_no_roles_check, in_whitelist_check -from bot.utils.time import humanize_delta, time_since +from bot.utils.time import TimestampFormats, discord_timestamp, humanize_delta  log = logging.getLogger(__name__) @@ -42,15 +44,29 @@ class Information(Cog):          return channel_counter      @staticmethod -    def get_member_counts(guild: Guild) -> Dict[str, int]: +    def join_role_stats(role_ids: list[int], guild: Guild, name: Optional[str] = None) -> dict[str, int]: +        """Return a dictionary with the number of `members` of each role given, and the `name` for this joined group.""" +        members = 0 +        for role_id in role_ids: +            if (role := guild.get_role(role_id)) is not None: +                members += len(role.members) +            else: +                raise NonExistentRoleError(role_id) +        return {name or role.name.title(): members} + +    @staticmethod +    def get_member_counts(guild: Guild) -> dict[str, int]:          """Return the total number of members for certain roles in `guild`.""" -        roles = ( -            guild.get_role(role_id) for role_id in ( -                constants.Roles.helpers, constants.Roles.moderators, constants.Roles.admins, -                constants.Roles.owners, constants.Roles.contributors, -            ) +        role_ids = [constants.Roles.helpers, constants.Roles.mod_team, constants.Roles.admins, +                    constants.Roles.owners, constants.Roles.contributors] + +        role_stats = {} +        for role_id in role_ids: +            role_stats.update(Information.join_role_stats([role_id], guild)) +        role_stats.update( +            Information.join_role_stats([constants.Roles.project_leads, constants.Roles.domain_leads], guild, "Leads")          ) -        return {role.name.title(): len(role.members) for role in roles} +        return role_stats      def get_extended_server_info(self, ctx: Context) -> str:          """Return additional server info only visible in moderation channels.""" @@ -154,7 +170,7 @@ class Information(Cog):          """Returns an embed full of server information."""          embed = Embed(colour=Colour.blurple(), title="Server Information") -        created = time_since(ctx.guild.created_at, precision="days") +        created = discord_timestamp(ctx.guild.created_at, TimestampFormats.RELATIVE)          region = ctx.guild.region          num_roles = len(ctx.guild.roles) - 1  # Exclude @everyone @@ -171,21 +187,21 @@ class Information(Cog):          online_presences = py_invite.approximate_presence_count          offline_presences = py_invite.approximate_member_count - online_presences          member_status = ( -            f"{constants.Emojis.status_online} {online_presences} " -            f"{constants.Emojis.status_offline} {offline_presences}" +            f"{constants.Emojis.status_online} {online_presences:,} " +            f"{constants.Emojis.status_offline} {offline_presences:,}"          ) -        embed.description = textwrap.dedent(f""" -            Created: {created} -            Voice region: {region}\ -            {features} -            Roles: {num_roles} -            Member status: {member_status} -        """) +        embed.description = ( +            f"Created: {created}" +            f"\nVoice region: {region}" +            f"{features}" +            f"\nRoles: {num_roles}" +            f"\nMember status: {member_status}" +        )          embed.set_thumbnail(url=ctx.guild.icon_url)          # Members -        total_members = ctx.guild.member_count +        total_members = f"{ctx.guild.member_count:,}"          member_counts = self.get_member_counts(ctx.guild)          member_info = "\n".join(f"{role}: {count}" for role, count in member_counts.items())          embed.add_field(name=f"Members: {total_members}", value=member_info) @@ -205,7 +221,7 @@ class Information(Cog):          await ctx.send(embed=embed)      @command(name="user", aliases=["user_info", "member", "member_info", "u"]) -    async def user_info(self, ctx: Context, user: FetchedMember = None) -> None: +    async def user_info(self, ctx: Context, user: MemberOrUser = None) -> None:          """Returns info about a user."""          if user is None:              user = ctx.author @@ -220,15 +236,16 @@ class Information(Cog):              embed = await self.create_user_embed(ctx, user)              await ctx.send(embed=embed) -    async def create_user_embed(self, ctx: Context, user: FetchedMember) -> Embed: +    async def create_user_embed(self, ctx: Context, user: MemberOrUser) -> Embed:          """Creates an embed containing information on the `user`."""          on_server = bool(ctx.guild.get_member(user.id)) -        created = time_since(user.created_at, max_units=3) +        created = discord_timestamp(user.created_at, TimestampFormats.RELATIVE)          name = str(user)          if on_server and user.nick:              name = f"{user.nick} ({name})" +        name = escape_markdown(name)          if user.public_flags.verified_bot:              name += f" {constants.Emojis.verified_bot}" @@ -242,8 +259,14 @@ class Information(Cog):                  badges.append(emoji)          if on_server: -            joined = time_since(user.joined_at, max_units=3) -            roles = ", ".join(role.mention for role in user.roles[1:]) +            if user.joined_at: +                joined = discord_timestamp(user.joined_at, TimestampFormats.RELATIVE) +            else: +                joined = "Unable to get join date" + +            # The 0 is for excluding the default @everyone role, +            # and the -1 is for reversing the order of the roles to highest to lowest in hierarchy. +            roles = ", ".join(role.mention for role in user.roles[:0:-1])              membership = {"Joined": joined, "Verified": not user.pending, "Roles": roles or None}              if not is_mod_channel(ctx.channel):                  membership.pop("Verified") @@ -290,7 +313,7 @@ class Information(Cog):          return embed -    async def basic_user_infraction_counts(self, user: FetchedMember) -> Tuple[str, str]: +    async def basic_user_infraction_counts(self, user: MemberOrUser) -> Tuple[str, str]:          """Gets the total and active infraction counts for the given `member`."""          infractions = await self.bot.api_client.get(              'bot/infractions', @@ -307,7 +330,7 @@ class Information(Cog):          return "Infractions", infraction_output -    async def expanded_user_infraction_counts(self, user: FetchedMember) -> Tuple[str, str]: +    async def expanded_user_infraction_counts(self, user: MemberOrUser) -> Tuple[str, str]:          """          Gets expanded infraction counts for the given `member`. @@ -348,7 +371,7 @@ class Information(Cog):          return "Infractions", "\n".join(infraction_output) -    async def user_nomination_counts(self, user: FetchedMember) -> Tuple[str, str]: +    async def user_nomination_counts(self, user: MemberOrUser) -> Tuple[str, str]:          """Gets the active and historical nomination counts for the given `member`."""          nominations = await self.bot.api_client.get(              'bot/nominations', @@ -373,7 +396,7 @@ class Information(Cog):          return "Nominations", "\n".join(output) -    async def user_messages(self, user: FetchedMember) -> Tuple[Union[bool, str], Tuple[str, str]]: +    async def user_messages(self, user: MemberOrUser) -> Tuple[Union[bool, str], Tuple[str, str]]:          """          Gets the amount of messages for `member`. diff --git a/bot/exts/info/pypi.py b/bot/exts/info/pypi.py index 2e42e7d6b..62498ce0b 100644 --- a/bot/exts/info/pypi.py +++ b/bot/exts/info/pypi.py @@ -2,13 +2,15 @@ import itertools  import logging  import random  import re +from contextlib import suppress -from discord import Embed +from discord import Embed, NotFound  from discord.ext.commands import Cog, Context, command  from discord.utils import escape_markdown  from bot.bot import Bot  from bot.constants import Colours, NEGATIVE_REPLIES, RedirectOutput +from bot.utils.messages import wait_for_deletion  URL = "https://pypi.org/pypi/{package}/json"  PYPI_ICON = "https://cdn.discordapp.com/emojis/766274397257334814.png" @@ -67,8 +69,15 @@ class PyPi(Cog):                      log.trace(f"Error when fetching PyPi package: {response.status}.")          if error: -            await ctx.send(embed=embed, delete_after=INVALID_INPUT_DELETE_DELAY) -            await ctx.message.delete(delay=INVALID_INPUT_DELETE_DELAY) +            error_message = await ctx.send(embed=embed) +            await wait_for_deletion(error_message, (ctx.author.id,), timeout=INVALID_INPUT_DELETE_DELAY) + +            # Make sure that we won't cause a ghost-ping by deleting the message +            if not (ctx.message.mentions or ctx.message.role_mentions): +                with suppress(NotFound): +                    await ctx.message.delete() +                    await error_message.delete() +          else:              await ctx.send(embed=embed) diff --git a/bot/exts/info/python_news.py b/bot/exts/info/python_news.py index 0ab5738a4..63eb4ac17 100644 --- a/bot/exts/info/python_news.py +++ b/bot/exts/info/python_news.py @@ -1,4 +1,5 @@  import logging +import re  import typing as t  from datetime import date, datetime @@ -72,6 +73,11 @@ class PythonNews(Cog):              if mail["name"].split("@")[0] in constants.PythonNews.mail_lists:                  self.webhook_names[mail["name"].split("@")[0]] = mail["display_name"] +    @staticmethod +    def escape_markdown(content: str) -> str: +        """Escape the markdown underlines and spoilers.""" +        return re.sub(r"[_|]", lambda match: "\\" + match[0], content) +      async def post_pep_news(self) -> None:          """Fetch new PEPs and when they don't have announcement in #python-news, create it."""          # Wait until everything is ready and http_session available @@ -103,7 +109,7 @@ class PythonNews(Cog):              # Build an embed and send a webhook              embed = discord.Embed(                  title=new["title"], -                description=new["summary"], +                description=self.escape_markdown(new["summary"]),                  timestamp=new_datetime,                  url=new["link"],                  colour=constants.Colours.soft_green @@ -167,13 +173,13 @@ class PythonNews(Cog):                  ):                      continue -                content = email_information["content"] +                content = self.escape_markdown(email_information["content"])                  link = THREAD_URL.format(id=thread["href"].split("/")[-2], list=maillist)                  # Build an embed and send a message to the webhook                  embed = discord.Embed(                      title=thread_information["subject"], -                    description=content[:500] + f"... [continue reading]({link})" if len(content) > 500 else content, +                    description=content[:1000] + f"... [continue reading]({link})" if len(content) > 1000 else content,                      timestamp=new_date,                      url=link,                      colour=constants.Colours.soft_green diff --git a/bot/exts/info/site.py b/bot/exts/info/site.py index fb5b99086..28eb558a6 100644 --- a/bot/exts/info/site.py +++ b/bot/exts/info/site.py @@ -9,7 +9,7 @@ from bot.pagination import LinePaginator  log = logging.getLogger(__name__) -PAGES_URL = f"{URLs.site_schema}{URLs.site}/pages" +BASE_URL = f"{URLs.site_schema}{URLs.site}"  class Site(Cog): @@ -43,7 +43,7 @@ class Site(Cog):      @site_group.command(name="resources", root_aliases=("resources", "resource"))      async def site_resources(self, ctx: Context) -> None:          """Info about the site's Resources page.""" -        learning_url = f"{PAGES_URL}/resources" +        learning_url = f"{BASE_URL}/resources"          embed = Embed(title="Resources")          embed.set_footer(text=f"{learning_url}") @@ -59,7 +59,7 @@ class Site(Cog):      @site_group.command(name="tools", root_aliases=("tools",))      async def site_tools(self, ctx: Context) -> None:          """Info about the site's Tools page.""" -        tools_url = f"{PAGES_URL}/resources/tools" +        tools_url = f"{BASE_URL}/resources/tools"          embed = Embed(title="Tools")          embed.set_footer(text=f"{tools_url}") @@ -74,7 +74,7 @@ class Site(Cog):      @site_group.command(name="help")      async def site_help(self, ctx: Context) -> None:          """Info about the site's Getting Help page.""" -        url = f"{PAGES_URL}/resources/guides/asking-good-questions" +        url = f"{BASE_URL}/pages/guides/pydis-guides/asking-good-questions/"          embed = Embed(title="Asking Good Questions")          embed.set_footer(text=url) @@ -90,7 +90,7 @@ class Site(Cog):      @site_group.command(name="faq", root_aliases=("faq",))      async def site_faq(self, ctx: Context) -> None:          """Info about the site's FAQ page.""" -        url = f"{PAGES_URL}/frequently-asked-questions" +        url = f"{BASE_URL}/pages/frequently-asked-questions"          embed = Embed(title="FAQ")          embed.set_footer(text=url) @@ -107,13 +107,13 @@ class Site(Cog):      @site_group.command(name="rules", aliases=("r", "rule"), root_aliases=("rules", "rule"))      async def site_rules(self, ctx: Context, rules: Greedy[int]) -> None:          """Provides a link to all rules or, if specified, displays specific rule(s).""" -        rules_embed = Embed(title='Rules', color=Colour.blurple(), url=f'{PAGES_URL}/rules') +        rules_embed = Embed(title='Rules', color=Colour.blurple(), url=f'{BASE_URL}/pages/rules')          if not rules:              # Rules were not submitted. Return the default description.              rules_embed.description = (                  "The rules and guidelines that apply to this community can be found on" -                f" our [rules page]({PAGES_URL}/rules). We expect" +                f" our [rules page]({BASE_URL}/pages/rules). We expect"                  " all members of the community to have read and understood these."              ) diff --git a/bot/exts/moderation/defcon.py b/bot/exts/moderation/defcon.py index dfb1afd19..6ac077b93 100644 --- a/bot/exts/moderation/defcon.py +++ b/bot/exts/moderation/defcon.py @@ -9,7 +9,7 @@ from typing import Optional, Union  from aioredis import RedisError  from async_rediscache import RedisCache  from dateutil.relativedelta import relativedelta -from discord import Colour, Embed, Member, User +from discord import Colour, Embed, Forbidden, Member, User  from discord.ext import tasks  from discord.ext.commands import Cog, Context, group, has_any_role @@ -19,7 +19,9 @@ from bot.converters import DurationDelta, Expiry  from bot.exts.moderation.modlog import ModLog  from bot.utils.messages import format_user  from bot.utils.scheduling import Scheduler -from bot.utils.time import humanize_delta, parse_duration_string, relativedelta_to_timedelta +from bot.utils.time import ( +    TimestampFormats, discord_timestamp, humanize_delta, parse_duration_string, relativedelta_to_timedelta +)  log = logging.getLogger(__name__) @@ -116,10 +118,12 @@ class Defcon(Cog):                  try:                      await member.send(REJECTION_MESSAGE.format(user=member.mention)) -                      message_sent = True +                except Forbidden: +                    log.debug(f"Cannot send DEFCON rejection DM to {member}: DMs disabled")                  except Exception: -                    log.exception(f"Unable to send rejection message to user: {member}") +                    # Broadly catch exceptions because DM isn't critical, but it's imperative to kick them. +                    log.exception(f"Error sending DEFCON rejection message to {member}")                  await member.kick(reason="DEFCON active, user is too new")                  self.bot.stats.incr("defcon.leaves") @@ -150,7 +154,7 @@ class Defcon(Cog):              colour=Colour.blurple(), title="DEFCON Status",              description=f"""                  **Threshold:** {humanize_delta(self.threshold) if self.threshold else "-"} -                **Expires in:** {humanize_delta(relativedelta(self.expiry, datetime.utcnow())) if self.expiry else "-"} +                **Expires:** {discord_timestamp(self.expiry, TimestampFormats.RELATIVE) if self.expiry else "-"}                  **Verification level:** {ctx.guild.verification_level.name}                  """          ) diff --git a/bot/exts/moderation/incidents.py b/bot/exts/moderation/incidents.py index 0e479d33f..561e0251e 100644 --- a/bot/exts/moderation/incidents.py +++ b/bot/exts/moderation/incidents.py @@ -143,7 +143,14 @@ async def add_signals(incident: discord.Message) -> None:              log.trace(f"Skipping emoji as it's already been placed: {signal_emoji}")          else:              log.trace(f"Adding reaction: {signal_emoji}") -            await incident.add_reaction(signal_emoji.value) +            try: +                await incident.add_reaction(signal_emoji.value) +            except discord.NotFound as e: +                if e.code != 10008: +                    raise + +                log.trace(f"Couldn't react with signal because message {incident.id} was deleted; skipping incident") +                return  class Incidents(Cog): @@ -288,14 +295,20 @@ class Incidents(Cog):          members_roles: t.Set[int] = {role.id for role in member.roles}          if not members_roles & ALLOWED_ROLES:  # Intersection is truthy on at least 1 common element              log.debug(f"Removing invalid reaction: user {member} is not permitted to send signals") -            await incident.remove_reaction(reaction, member) +            try: +                await incident.remove_reaction(reaction, member) +            except discord.NotFound: +                log.trace("Couldn't remove reaction because the reaction or its message was deleted")              return          try:              signal = Signal(reaction)          except ValueError:              log.debug(f"Removing invalid reaction: emoji {reaction} is not a valid signal") -            await incident.remove_reaction(reaction, member) +            try: +                await incident.remove_reaction(reaction, member) +            except discord.NotFound: +                log.trace("Couldn't remove reaction because the reaction or its message was deleted")              return          log.trace(f"Received signal: {signal}") @@ -313,7 +326,10 @@ class Incidents(Cog):          confirmation_task = self.make_confirmation_task(incident, timeout)          log.trace("Deleting original message") -        await incident.delete() +        try: +            await incident.delete() +        except discord.NotFound: +            log.trace("Couldn't delete message because it was already deleted")          log.trace(f"Awaiting deletion confirmation: {timeout=} seconds")          try: diff --git a/bot/exts/moderation/infraction/_scheduler.py b/bot/exts/moderation/infraction/_scheduler.py index c2fd959f7..6ba4e74e9 100644 --- a/bot/exts/moderation/infraction/_scheduler.py +++ b/bot/exts/moderation/infraction/_scheduler.py @@ -13,8 +13,8 @@ from bot import constants  from bot.api import ResponseCodeError  from bot.bot import Bot  from bot.constants import Colours +from bot.converters import MemberOrUser  from bot.exts.moderation.infraction import _utils -from bot.exts.moderation.infraction._utils import UserSnowflake  from bot.exts.moderation.modlog import ModLog  from bot.utils import messages, scheduling, time  from bot.utils.channel import is_mod_channel @@ -115,7 +115,7 @@ class InfractionScheduler:          self,          ctx: Context,          infraction: _utils.Infraction, -        user: UserSnowflake, +        user: MemberOrUser,          action_coro: t.Optional[t.Awaitable] = None,          user_reason: t.Optional[str] = None,          additional_info: str = "", @@ -165,17 +165,10 @@ class InfractionScheduler:              dm_result = f"{constants.Emojis.failmail} "              dm_log_text = "\nDM: **Failed**" -            # Sometimes user is a discord.Object; make it a proper user. -            try: -                if not isinstance(user, (discord.Member, discord.User)): -                    user = await self.bot.fetch_user(user.id) -            except discord.HTTPException as e: -                log.error(f"Failed to DM {user.id}: could not fetch user (status {e.status})") -            else: -                # Accordingly display whether the user was successfully notified via DM. -                if await _utils.notify_infraction(user, infr_type.replace("_", " ").title(), expiry, user_reason, icon): -                    dm_result = ":incoming_envelope: " -                    dm_log_text = "\nDM: Sent" +            # Accordingly display whether the user was successfully notified via DM. +            if await _utils.notify_infraction(user, infr_type.replace("_", " ").title(), expiry, user_reason, icon): +                dm_result = ":incoming_envelope: " +                dm_log_text = "\nDM: Sent"          end_msg = ""          if infraction["actor"] == self.bot.user.id: @@ -264,7 +257,7 @@ class InfractionScheduler:              self,              ctx: Context,              infr_type: str, -            user: UserSnowflake, +            user: MemberOrUser,              *,              send_msg: bool = True,              notify: bool = True diff --git a/bot/exts/moderation/infraction/_utils.py b/bot/exts/moderation/infraction/_utils.py index dd427e413..b20ef1d06 100644 --- a/bot/exts/moderation/infraction/_utils.py +++ b/bot/exts/moderation/infraction/_utils.py @@ -7,6 +7,7 @@ from discord.ext.commands import Context  from bot.api import ResponseCodeError  from bot.constants import Colours, Icons +from bot.converters import MemberOrUser  from bot.errors import InvalidInfractedUserError  log = logging.getLogger(__name__) @@ -24,8 +25,6 @@ INFRACTION_ICONS = {  RULES_URL = "https://pythondiscord.com/pages/rules"  # Type aliases -UserObject = t.Union[discord.Member, discord.User] -UserSnowflake = t.Union[UserObject, discord.Object]  Infraction = t.Dict[str, t.Union[str, int, bool]]  APPEAL_EMAIL = "[email protected]" @@ -45,7 +44,7 @@ INFRACTION_DESCRIPTION_TEMPLATE = (  ) -async def post_user(ctx: Context, user: UserSnowflake) -> t.Optional[dict]: +async def post_user(ctx: Context, user: MemberOrUser) -> t.Optional[dict]:      """      Create a new user in the database. @@ -53,14 +52,11 @@ async def post_user(ctx: Context, user: UserSnowflake) -> t.Optional[dict]:      """      log.trace(f"Attempting to add user {user.id} to the database.") -    if not isinstance(user, (discord.Member, discord.User)): -        log.debug("The user being added to the DB is not a Member or User object.") -      payload = { -        'discriminator': int(getattr(user, 'discriminator', 0)), +        'discriminator': int(user.discriminator),          'id': user.id,          'in_guild': False, -        'name': getattr(user, 'name', 'Name unknown'), +        'name': user.name,          'roles': []      } @@ -75,7 +71,7 @@ async def post_user(ctx: Context, user: UserSnowflake) -> t.Optional[dict]:  async def post_infraction(          ctx: Context, -        user: UserSnowflake, +        user: MemberOrUser,          infr_type: str,          reason: str,          expires_at: datetime = None, @@ -118,7 +114,7 @@ async def post_infraction(  async def get_active_infraction(          ctx: Context, -        user: UserSnowflake, +        user: MemberOrUser,          infr_type: str,          send_msg: bool = True  ) -> t.Optional[dict]: @@ -158,7 +154,7 @@ async def send_active_infraction_message(ctx: Context, infraction: Infraction) -  async def notify_infraction( -        user: UserObject, +        user: MemberOrUser,          infr_type: str,          expires_at: t.Optional[str] = None,          reason: t.Optional[str] = None, @@ -169,13 +165,13 @@ async def notify_infraction(      text = INFRACTION_DESCRIPTION_TEMPLATE.format(          type=infr_type.title(), -        expires=f"{expires_at} UTC" if expires_at else "N/A", +        expires=expires_at or "N/A",          reason=reason or "No reason provided."      )      # For case when other fields than reason is too long and this reach limit, then force-shorten string -    if len(text) > 2048: -        text = f"{text[:2045]}..." +    if len(text) > 4096: +        text = f"{text[:4093]}..."      embed = discord.Embed(          description=text, @@ -194,7 +190,7 @@ async def notify_infraction(  async def notify_pardon( -        user: UserObject, +        user: MemberOrUser,          title: str,          content: str,          icon_url: str = Icons.user_verified @@ -212,7 +208,7 @@ async def notify_pardon(      return await send_private_embed(user, embed) -async def send_private_embed(user: UserObject, embed: discord.Embed) -> bool: +async def send_private_embed(user: MemberOrUser, embed: discord.Embed) -> bool:      """      A helper method for sending an embed to a user's DMs. diff --git a/bot/exts/moderation/infraction/infractions.py b/bot/exts/moderation/infraction/infractions.py index 0df5fb60b..2f9083c29 100644 --- a/bot/exts/moderation/infraction/infractions.py +++ b/bot/exts/moderation/infraction/infractions.py @@ -10,11 +10,10 @@ from discord.ext.commands import Context, command  from bot import constants  from bot.bot import Bot  from bot.constants import Event -from bot.converters import Duration, Expiry, FetchedMember +from bot.converters import Duration, Expiry, MemberOrUser  from bot.decorators import respect_role_hierarchy  from bot.exts.moderation.infraction import _utils  from bot.exts.moderation.infraction._scheduler import InfractionScheduler -from bot.exts.moderation.infraction._utils import UserSnowflake  from bot.utils.messages import format_user  log = logging.getLogger(__name__) @@ -54,7 +53,7 @@ class Infractions(InfractionScheduler, commands.Cog):      # region: Permanent infractions      @command() -    async def warn(self, ctx: Context, user: FetchedMember, *, reason: t.Optional[str] = None) -> None: +    async def warn(self, ctx: Context, user: MemberOrUser, *, reason: t.Optional[str] = None) -> None:          """Warn a user for the given reason."""          if not isinstance(user, Member):              await ctx.send(":x: The user doesn't appear to be on the server.") @@ -67,7 +66,7 @@ class Infractions(InfractionScheduler, commands.Cog):          await self.apply_infraction(ctx, infraction, user)      @command() -    async def kick(self, ctx: Context, user: FetchedMember, *, reason: t.Optional[str] = None) -> None: +    async def kick(self, ctx: Context, user: MemberOrUser, *, reason: t.Optional[str] = None) -> None:          """Kick a user for the given reason."""          if not isinstance(user, Member):              await ctx.send(":x: The user doesn't appear to be on the server.") @@ -79,7 +78,7 @@ class Infractions(InfractionScheduler, commands.Cog):      async def ban(          self,          ctx: Context, -        user: FetchedMember, +        user: MemberOrUser,          duration: t.Optional[Expiry] = None,          *,          reason: t.Optional[str] = None @@ -95,7 +94,7 @@ class Infractions(InfractionScheduler, commands.Cog):      async def purgeban(          self,          ctx: Context, -        user: FetchedMember, +        user: MemberOrUser,          duration: t.Optional[Expiry] = None,          *,          reason: t.Optional[str] = None @@ -111,7 +110,7 @@ class Infractions(InfractionScheduler, commands.Cog):      async def voiceban(          self,          ctx: Context, -        user: FetchedMember, +        user: MemberOrUser,          duration: t.Optional[Expiry] = None,          *,          reason: t.Optional[str] @@ -129,7 +128,7 @@ class Infractions(InfractionScheduler, commands.Cog):      @command(aliases=["mute"])      async def tempmute(          self, ctx: Context, -        user: FetchedMember, +        user: MemberOrUser,          duration: t.Optional[Expiry] = None,          *,          reason: t.Optional[str] = None @@ -163,7 +162,7 @@ class Infractions(InfractionScheduler, commands.Cog):      async def tempban(          self,          ctx: Context, -        user: FetchedMember, +        user: MemberOrUser,          duration: Expiry,          *,          reason: t.Optional[str] = None @@ -189,7 +188,7 @@ class Infractions(InfractionScheduler, commands.Cog):      async def tempvoiceban(              self,              ctx: Context, -            user: FetchedMember, +            user: MemberOrUser,              duration: Expiry,              *,              reason: t.Optional[str] @@ -215,7 +214,7 @@ class Infractions(InfractionScheduler, commands.Cog):      # region: Permanent shadow infractions      @command(hidden=True) -    async def note(self, ctx: Context, user: FetchedMember, *, reason: t.Optional[str] = None) -> None: +    async def note(self, ctx: Context, user: MemberOrUser, *, reason: t.Optional[str] = None) -> None:          """Create a private note for a user with the given reason without notifying the user."""          infraction = await _utils.post_infraction(ctx, user, "note", reason, hidden=True, active=False)          if infraction is None: @@ -224,7 +223,7 @@ class Infractions(InfractionScheduler, commands.Cog):          await self.apply_infraction(ctx, infraction, user)      @command(hidden=True, aliases=['shadowban', 'sban']) -    async def shadow_ban(self, ctx: Context, user: FetchedMember, *, reason: t.Optional[str] = None) -> None: +    async def shadow_ban(self, ctx: Context, user: MemberOrUser, *, reason: t.Optional[str] = None) -> None:          """Permanently ban a user for the given reason without notifying the user."""          await self.apply_ban(ctx, user, reason, hidden=True) @@ -235,7 +234,7 @@ class Infractions(InfractionScheduler, commands.Cog):      async def shadow_tempban(          self,          ctx: Context, -        user: FetchedMember, +        user: MemberOrUser,          duration: Expiry,          *,          reason: t.Optional[str] = None @@ -261,17 +260,17 @@ class Infractions(InfractionScheduler, commands.Cog):      # region: Remove infractions (un- commands)      @command() -    async def unmute(self, ctx: Context, user: FetchedMember) -> None: +    async def unmute(self, ctx: Context, user: MemberOrUser) -> None:          """Prematurely end the active mute infraction for the user."""          await self.pardon_infraction(ctx, "mute", user)      @command() -    async def unban(self, ctx: Context, user: FetchedMember) -> None: +    async def unban(self, ctx: Context, user: MemberOrUser) -> None:          """Prematurely end the active ban infraction for the user."""          await self.pardon_infraction(ctx, "ban", user)      @command(aliases=("uvban",)) -    async def unvoiceban(self, ctx: Context, user: FetchedMember) -> None: +    async def unvoiceban(self, ctx: Context, user: MemberOrUser) -> None:          """Prematurely end the active voice ban infraction for the user."""          await self.pardon_infraction(ctx, "voice_ban", user) @@ -331,7 +330,7 @@ class Infractions(InfractionScheduler, commands.Cog):      async def apply_ban(          self,          ctx: Context, -        user: UserSnowflake, +        user: MemberOrUser,          reason: t.Optional[str],          purge_days: t.Optional[int] = 0,          **kwargs @@ -387,7 +386,7 @@ class Infractions(InfractionScheduler, commands.Cog):          await bb_cog.apply_unwatch(ctx, user, bb_reason, send_message=False)      @respect_role_hierarchy(member_arg=2) -    async def apply_voice_ban(self, ctx: Context, user: UserSnowflake, reason: t.Optional[str], **kwargs) -> None: +    async def apply_voice_ban(self, ctx: Context, user: MemberOrUser, reason: t.Optional[str], **kwargs) -> None:          """Apply a voice ban infraction with kwargs passed to `post_infraction`."""          if await _utils.get_active_infraction(ctx, user, "voice_ban"):              return diff --git a/bot/exts/moderation/infraction/management.py b/bot/exts/moderation/infraction/management.py index b3783cd60..641ad0410 100644 --- a/bot/exts/moderation/infraction/management.py +++ b/bot/exts/moderation/infraction/management.py @@ -3,19 +3,22 @@ import textwrap  import typing as t  from datetime import datetime +import dateutil.parser  import discord +from dateutil.relativedelta import relativedelta  from discord.ext import commands  from discord.ext.commands import Context  from discord.utils import escape_markdown  from bot import constants  from bot.bot import Bot -from bot.converters import Expiry, Infraction, Snowflake, UserMention, allowed_strings, proxy_user +from bot.converters import Expiry, Infraction, MemberOrUser, Snowflake, UserMentionOrID, allowed_strings  from bot.exts.moderation.infraction.infractions import Infractions  from bot.exts.moderation.modlog import ModLog  from bot.pagination import LinePaginator  from bot.utils import messages, time  from bot.utils.channel import is_mod_channel +from bot.utils.time import humanize_delta, until_expiration  log = logging.getLogger(__name__) @@ -78,7 +81,7 @@ class ModManagement(commands.Cog):          """          old_reason = infraction["reason"] -        if old_reason is not None: +        if old_reason is not None and reason is not None:              add_period = not old_reason.endswith((".", "!", "?"))              reason = old_reason + (". " if add_period else " ") + reason @@ -164,8 +167,8 @@ class ModManagement(commands.Cog):                  self.infractions_cog.schedule_expiration(new_infraction)              log_text += f""" -                Previous expiry: {infraction['expires_at'] or "Permanent"} -                New expiry: {new_infraction['expires_at'] or "Permanent"} +                Previous expiry: {until_expiration(infraction['expires_at']) or "Permanent"} +                New expiry: {until_expiration(new_infraction['expires_at']) or "Permanent"}              """.rstrip()          changes = ' & '.join(confirm_messages) @@ -198,29 +201,34 @@ class ModManagement(commands.Cog):      # region: Search infractions      @infraction_group.group(name="search", aliases=('s',), invoke_without_command=True) -    async def infraction_search_group(self, ctx: Context, query: t.Union[UserMention, Snowflake, str]) -> None: +    async def infraction_search_group(self, ctx: Context, query: t.Union[UserMentionOrID, Snowflake, str]) -> None:          """Searches for infractions in the database."""          if isinstance(query, int):              await self.search_user(ctx, discord.Object(query)) -        else: +        elif isinstance(query, str):              await self.search_reason(ctx, query) +        else: +            await self.search_user(ctx, query)      @infraction_search_group.command(name="user", aliases=("member", "id")) -    async def search_user(self, ctx: Context, user: t.Union[discord.User, proxy_user]) -> None: +    async def search_user(self, ctx: Context, user: t.Union[MemberOrUser, discord.Object]) -> None:          """Search for infractions by member."""          infraction_list = await self.bot.api_client.get(              'bot/infractions/expanded',              params={'user__id': str(user.id)}          ) -        user = self.bot.get_user(user.id) -        if not user and infraction_list: -            # Use the user data retrieved from the DB for the username. -            user = infraction_list[0]["user"] -            user = escape_markdown(user["name"]) + f"#{user['discriminator']:04}" +        if isinstance(user, (discord.Member, discord.User)): +            user_str = escape_markdown(str(user)) +        else: +            if infraction_list: +                user = infraction_list[0]["user"] +                user_str = escape_markdown(user["name"]) + f"#{user['discriminator']:04}" +            else: +                user_str = str(user.id)          embed = discord.Embed( -            title=f"Infractions for {user} ({len(infraction_list)} total)", +            title=f"Infractions for {user_str} ({len(infraction_list)} total)",              colour=discord.Colour.orange()          )          await self.send_infraction_list(ctx, embed, infraction_list) @@ -288,10 +296,11 @@ class ModManagement(commands.Cog):              remaining = "Inactive"          if expires_at is None: -            expires = "*Permanent*" +            duration = "*Permanent*"          else: -            date_from = datetime.strptime(created, time.INFRACTION_FORMAT) -            expires = time.format_infraction_with_duration(expires_at, date_from) +            date_from = datetime.fromtimestamp(float(time.DISCORD_TIMESTAMP_REGEX.match(created).group(1))) +            date_to = dateutil.parser.isoparse(expires_at).replace(tzinfo=None) +            duration = humanize_delta(relativedelta(date_to, date_from))          lines = textwrap.dedent(f"""              {"**===============**" if active else "==============="} @@ -300,8 +309,8 @@ class ModManagement(commands.Cog):              Type: **{infraction["type"]}**              Shadow: {infraction["hidden"]}              Created: {created} -            Expires: {expires} -            Remaining: {remaining} +            Expires: {remaining} +            Duration: {duration}              Actor: <@{infraction["actor"]["id"]}>              ID: `{infraction["id"]}`              Reason: {infraction["reason"] or "*None*"} diff --git a/bot/exts/moderation/metabase.py b/bot/exts/moderation/metabase.py index db5f04d83..3b454ab18 100644 --- a/bot/exts/moderation/metabase.py +++ b/bot/exts/moderation/metabase.py @@ -42,6 +42,25 @@ class Metabase(Cog):          self.init_task = self.bot.loop.create_task(self.init_cog()) +    async def cog_command_error(self, ctx: Context, error: Exception) -> None: +        """Handle ClientResponseError errors locally to invalidate token if needed.""" +        if not isinstance(error.original, ClientResponseError): +            return + +        if error.original.status == 403: +            # User doesn't have access to the given question +            log.warning(f"Failed to auth with Metabase for {error.original.url}.") +            await ctx.send(f":x: {ctx.author.mention} Failed to auth with Metabase for that question.") +        elif error.original.status == 404: +            await ctx.send(f":x: {ctx.author.mention} That question could not be found.") +        else: +            # User credentials are invalid, or the refresh failed. +            # Delete the expiry time, to force a refresh on next startup. +            await self.session_info.delete("session_expiry") +            log.exception("Session token is invalid or refresh failed.") +            await ctx.send(f":x: {ctx.author.mention} Session token is invalid or refresh failed.") +        error.handled = True +      async def init_cog(self) -> None:          """Initialise the metabase session."""          expiry_time = await self.session_info.get("session_expiry") @@ -65,7 +84,7 @@ class Metabase(Cog):              "username": MetabaseConfig.username,              "password": MetabaseConfig.password          } -        async with self.bot.http_session.post(f"{MetabaseConfig.url}/session", json=data) as resp: +        async with self.bot.http_session.post(f"{MetabaseConfig.base_url}/api/session", json=data) as resp:              json_data = await resp.json()              self.session_token = json_data.get("id") @@ -86,7 +105,7 @@ class Metabase(Cog):          """A group of commands for interacting with metabase."""          await ctx.send_help(ctx.command) -    @metabase_group.command(name="extract") +    @metabase_group.command(name="extract", aliases=("export",))      async def metabase_extract(          self,          ctx: Context, @@ -106,48 +125,50 @@ class Metabase(Cog):          Valid extensions are: csv and json.          """ -        async with ctx.typing(): - -            # Make sure we have a session token before running anything -            await self.init_task - -            url = f"{MetabaseConfig.url}/card/{question_id}/query/{extension}" -            try: -                async with self.bot.http_session.post(url, headers=self.headers, raise_for_status=True) as resp: -                    if extension == "csv": -                        out = await resp.text() -                        # Save the output for use with int e -                        self.exports[question_id] = list(csv.DictReader(StringIO(out))) - -                    elif extension == "json": -                        out = await resp.json() -                        # Save the output for use with int e -                        self.exports[question_id] = out - -                        # Format it nicely for human eyes -                        out = json.dumps(out, indent=4, sort_keys=True) -            except ClientResponseError as e: -                if e.status == 403: -                    # User doesn't have access to the given question -                    log.warning(f"Failed to auth with Metabase for question {question_id}.") -                    await ctx.send(f":x: {ctx.author.mention} Failed to auth with Metabase for that question.") -                else: -                    # User credentials are invalid, or the refresh failed. -                    # Delete the expiry time, to force a refresh on next startup. -                    await self.session_info.delete("session_expiry") -                    log.exception("Session token is invalid or refresh failed.") -                    await ctx.send(f":x: {ctx.author.mention} Session token is invalid or refresh failed.") -                return - -            paste_link = await send_to_paste_service(out, extension=extension) -            if paste_link: -                message = f":+1: {ctx.author.mention} Here's your link: {paste_link}" -            else: -                message = f":x: {ctx.author.mention} Link service is unavailible." -            await ctx.send( -                f"{message}\nYou can also access this data within internal eval by doing: " -                f"`bot.get_cog('Metabase').exports[{question_id}]`" -            ) +        await ctx.trigger_typing() + +        # Make sure we have a session token before running anything +        await self.init_task + +        url = f"{MetabaseConfig.base_url}/api/card/{question_id}/query/{extension}" + +        async with self.bot.http_session.post(url, headers=self.headers, raise_for_status=True) as resp: +            if extension == "csv": +                out = await resp.text(encoding="utf-8") +                # Save the output for use with int e +                self.exports[question_id] = list(csv.DictReader(StringIO(out))) + +            elif extension == "json": +                out = await resp.json(encoding="utf-8") +                # Save the output for use with int e +                self.exports[question_id] = out + +                # Format it nicely for human eyes +                out = json.dumps(out, indent=4, sort_keys=True) + +        paste_link = await send_to_paste_service(out, extension=extension) +        if paste_link: +            message = f":+1: {ctx.author.mention} Here's your link: {paste_link}" +        else: +            message = f":x: {ctx.author.mention} Link service is unavailible." +        await ctx.send( +            f"{message}\nYou can also access this data within internal eval by doing: " +            f"`bot.get_cog('Metabase').exports[{question_id}]`" +        ) + +    @metabase_group.command(name="publish", aliases=("share",)) +    async def metabase_publish(self, ctx: Context, question_id: int) -> None: +        """Publically shares the given question and posts the link.""" +        await ctx.trigger_typing() +        # Make sure we have a session token before running anything +        await self.init_task + +        url = f"{MetabaseConfig.base_url}/api/card/{question_id}/public_link" + +        async with self.bot.http_session.post(url, headers=self.headers, raise_for_status=True) as resp: +            response_json = await resp.json(encoding="utf-8") +            sharing_url = f"{MetabaseConfig.base_url}/public/question/{response_json['uuid']}" +            await ctx.send(f":+1: {ctx.author.mention} Here's your sharing link: {sharing_url}")      # This cannot be static (must have a __func__ attribute).      async def cog_check(self, ctx: Context) -> bool: diff --git a/bot/exts/moderation/modlog.py b/bot/exts/moderation/modlog.py index be65ade6e..be2245650 100644 --- a/bot/exts/moderation/modlog.py +++ b/bot/exts/moderation/modlog.py @@ -99,7 +99,7 @@ class ModLog(Cog, name="ModLog"):          """Generate log embed and send to logging channel."""          # Truncate string directly here to avoid removing newlines          embed = discord.Embed( -            description=text[:2045] + "..." if len(text) > 2048 else text +            description=text[:4093] + "..." if len(text) > 4096 else text          )          if title and icon_url: @@ -564,7 +564,7 @@ class ModLog(Cog, name="ModLog"):          # Shorten the message content if necessary          content = message.clean_content -        remaining_chars = 2040 - len(response) +        remaining_chars = 4090 - len(response)          if len(content) > remaining_chars:              botlog_url = await self.upload_log(messages=[message], actor_id=message.author.id) diff --git a/bot/exts/moderation/modpings.py b/bot/exts/moderation/modpings.py index 1ad5005de..29a5c1c8e 100644 --- a/bot/exts/moderation/modpings.py +++ b/bot/exts/moderation/modpings.py @@ -44,7 +44,7 @@ class ModPings(Cog):          log.trace("Applying the moderators role to the mod team where necessary.")          for mod in mod_team.members:              if mod in pings_on:  # Make sure that on-duty mods aren't in the cache. -                if mod in pings_off: +                if mod.id in pings_off:                      await self.pings_off_mods.delete(mod.id)                  continue @@ -59,6 +59,7 @@ class ModPings(Cog):          """Reapply the moderator's role to the given moderator."""          log.trace(f"Re-applying role to mod with ID {mod.id}.")          await mod.add_roles(self.moderators_role, reason="Pings off period expired.") +        await self.pings_off_mods.delete(mod.id)      @group(name='modpings', aliases=('modping',), invoke_without_command=True)      @has_any_role(*MODERATION_ROLES) diff --git a/bot/exts/moderation/stream.py b/bot/exts/moderation/stream.py index fd856a7f4..07ee4099e 100644 --- a/bot/exts/moderation/stream.py +++ b/bot/exts/moderation/stream.py @@ -13,7 +13,7 @@ from bot.constants import Colours, Emojis, Guild, MODERATION_ROLES, Roles, STAFF  from bot.converters import Expiry  from bot.pagination import LinePaginator  from bot.utils.scheduling import Scheduler -from bot.utils.time import format_infraction_with_duration +from bot.utils.time import discord_timestamp, format_infraction_with_duration  log = logging.getLogger(__name__) @@ -134,16 +134,7 @@ class Stream(commands.Cog):          await member.add_roles(discord.Object(Roles.video), reason="Temporary streaming access granted") -        # Use embed as embed timestamps do timezone conversions. -        embed = discord.Embed( -            description=f"{Emojis.check_mark} {member.mention} can now stream.", -            colour=Colours.soft_green -        ) -        embed.set_footer(text=f"Streaming permission has been given to {member} until") -        embed.timestamp = duration - -        # Mention in content as mentions in embeds don't ping -        await ctx.send(content=member.mention, embed=embed) +        await ctx.send(f"{Emojis.check_mark} {member.mention} can now stream until {discord_timestamp(duration)}.")          # Convert here for nicer logging          revoke_time = format_infraction_with_duration(str(duration)) diff --git a/bot/exts/moderation/watchchannels/_watchchannel.py b/bot/exts/moderation/watchchannels/_watchchannel.py index 9f26c34f2..146426569 100644 --- a/bot/exts/moderation/watchchannels/_watchchannel.py +++ b/bot/exts/moderation/watchchannels/_watchchannel.py @@ -295,7 +295,7 @@ class WatchChannel(metaclass=CogABCMeta):          footer = f"Added {time_delta} by {actor} | Reason: {reason}"          embed = Embed(description=f"{msg.author.mention} {message_jump}") -        embed.set_footer(text=textwrap.shorten(footer, width=128, placeholder="...")) +        embed.set_footer(text=textwrap.shorten(footer, width=256, placeholder="..."))          await self.webhook_send(embed=embed, username=msg.author.display_name, avatar_url=msg.author.avatar_url) diff --git a/bot/exts/moderation/watchchannels/bigbrother.py b/bot/exts/moderation/watchchannels/bigbrother.py index c6ee844ef..3aa253fea 100644 --- a/bot/exts/moderation/watchchannels/bigbrother.py +++ b/bot/exts/moderation/watchchannels/bigbrother.py @@ -6,7 +6,7 @@ from discord.ext.commands import Cog, Context, group, has_any_role  from bot.bot import Bot  from bot.constants import Channels, MODERATION_ROLES, Webhooks -from bot.converters import FetchedMember +from bot.converters import MemberOrUser  from bot.exts.moderation.infraction._utils import post_infraction  from bot.exts.moderation.watchchannels._watchchannel import WatchChannel @@ -60,7 +60,7 @@ class BigBrother(WatchChannel, Cog, name="Big Brother"):      @bigbrother_group.command(name='watch', aliases=('w',), root_aliases=('watch',))      @has_any_role(*MODERATION_ROLES) -    async def watch_command(self, ctx: Context, user: FetchedMember, *, reason: str) -> None: +    async def watch_command(self, ctx: Context, user: MemberOrUser, *, reason: str) -> None:          """          Relay messages sent by the given `user` to the `#big-brother` channel. @@ -71,11 +71,11 @@ class BigBrother(WatchChannel, Cog, name="Big Brother"):      @bigbrother_group.command(name='unwatch', aliases=('uw',), root_aliases=('unwatch',))      @has_any_role(*MODERATION_ROLES) -    async def unwatch_command(self, ctx: Context, user: FetchedMember, *, reason: str) -> None: +    async def unwatch_command(self, ctx: Context, user: MemberOrUser, *, reason: str) -> None:          """Stop relaying messages by the given `user`."""          await self.apply_unwatch(ctx, user, reason) -    async def apply_watch(self, ctx: Context, user: FetchedMember, reason: str) -> None: +    async def apply_watch(self, ctx: Context, user: MemberOrUser, reason: str) -> None:          """          Add `user` to watched users and apply a watch infraction with `reason`. @@ -94,7 +94,7 @@ class BigBrother(WatchChannel, Cog, name="Big Brother"):              await ctx.send(f":x: {user} is already being watched.")              return -        # FetchedUser instances don't have a roles attribute +        # discord.User instances don't have a roles attribute          if hasattr(user, "roles") and any(role.id in MODERATION_ROLES for role in user.roles):              await ctx.send(f":x: I'm sorry {ctx.author}, I'm afraid I can't do that. I must be kind to my masters.")              return @@ -125,7 +125,7 @@ class BigBrother(WatchChannel, Cog, name="Big Brother"):          await ctx.send(msg) -    async def apply_unwatch(self, ctx: Context, user: FetchedMember, reason: str, send_message: bool = True) -> None: +    async def apply_unwatch(self, ctx: Context, user: MemberOrUser, reason: str, send_message: bool = True) -> None:          """          Remove `user` from watched users and mark their infraction as inactive with `reason`. diff --git a/bot/exts/recruitment/talentpool/_cog.py b/bot/exts/recruitment/talentpool/_cog.py index 03326cab2..5c1a1cd3f 100644 --- a/bot/exts/recruitment/talentpool/_cog.py +++ b/bot/exts/recruitment/talentpool/_cog.py @@ -5,18 +5,20 @@ from io import StringIO  from typing import Union  import discord -from discord import Color, Embed, Member, PartialMessage, RawReactionActionEvent, User +from async_rediscache import RedisCache +from discord import Color, Embed, Member, PartialMessage, RawReactionActionEvent  from discord.ext.commands import Cog, Context, group, has_any_role  from bot.api import ResponseCodeError  from bot.bot import Bot -from bot.constants import Channels, Emojis, Guild, MODERATION_ROLES, STAFF_ROLES, Webhooks -from bot.converters import FetchedMember +from bot.constants import Channels, Emojis, Guild, MODERATION_ROLES, Roles, STAFF_ROLES, Webhooks +from bot.converters import MemberOrUser  from bot.exts.moderation.watchchannels._watchchannel import WatchChannel  from bot.exts.recruitment.talentpool._review import Reviewer  from bot.pagination import LinePaginator  from bot.utils import time +AUTOREVIEW_ENABLED_KEY = "autoreview_enabled"  REASON_MAX_CHARS = 1000  log = logging.getLogger(__name__) @@ -25,6 +27,10 @@ log = logging.getLogger(__name__)  class TalentPool(WatchChannel, Cog, name="Talentpool"):      """Relays messages of helper candidates to a watch channel to observe them.""" +    # RedisCache[str, bool] +    # Can contain a single key, "autoreview_enabled", with the value a bool indicating if autoreview is enabled. +    talentpool_settings = RedisCache() +      def __init__(self, bot: Bot) -> None:          super().__init__(              bot, @@ -37,7 +43,18 @@ class TalentPool(WatchChannel, Cog, name="Talentpool"):          )          self.reviewer = Reviewer(self.__class__.__name__, bot, self) -        self.bot.loop.create_task(self.reviewer.reschedule_reviews()) +        self.bot.loop.create_task(self.schedule_autoreviews()) + +    async def schedule_autoreviews(self) -> None: +        """Reschedule reviews for active nominations if autoreview is enabled.""" +        if await self.autoreview_enabled(): +            await self.reviewer.reschedule_reviews() +        else: +            self.log.trace("Not scheduling reviews as autoreview is disabled.") + +    async def autoreview_enabled(self) -> bool: +        """Return whether automatic posting of nomination reviews is enabled.""" +        return await self.talentpool_settings.get(AUTOREVIEW_ENABLED_KEY, True)      @group(name='talentpool', aliases=('tp', 'talent', 'nomination', 'n'), invoke_without_command=True)      @has_any_role(*MODERATION_ROLES) @@ -45,6 +62,50 @@ class TalentPool(WatchChannel, Cog, name="Talentpool"):          """Highlights the activity of helper nominees by relaying their messages to the talent pool channel."""          await ctx.send_help(ctx.command) +    @nomination_group.group(name="autoreview", aliases=("ar",), invoke_without_command=True) +    @has_any_role(*MODERATION_ROLES) +    async def nomination_autoreview_group(self, ctx: Context) -> None: +        """Commands for enabling or disabling autoreview.""" +        await ctx.send_help(ctx.command) + +    @nomination_autoreview_group.command(name="enable", aliases=("on",)) +    @has_any_role(Roles.admins) +    async def autoreview_enable(self, ctx: Context) -> None: +        """ +        Enable automatic posting of reviews. + +        This will post reviews up to one day overdue. Older nominations can be +        manually reviewed with the `tp post_review <user_id>` command. +        """ +        if await self.autoreview_enabled(): +            await ctx.send(":x: Autoreview is already enabled") +            return + +        await self.talentpool_settings.set(AUTOREVIEW_ENABLED_KEY, True) +        await self.reviewer.reschedule_reviews() +        await ctx.send(":white_check_mark: Autoreview enabled") + +    @nomination_autoreview_group.command(name="disable", aliases=("off",)) +    @has_any_role(Roles.admins) +    async def autoreview_disable(self, ctx: Context) -> None: +        """Disable automatic posting of reviews.""" +        if not await self.autoreview_enabled(): +            await ctx.send(":x: Autoreview is already disabled") +            return + +        await self.talentpool_settings.set(AUTOREVIEW_ENABLED_KEY, False) +        self.reviewer.cancel_all() +        await ctx.send(":white_check_mark: Autoreview disabled") + +    @nomination_autoreview_group.command(name="status") +    @has_any_role(*MODERATION_ROLES) +    async def autoreview_status(self, ctx: Context) -> None: +        """Show whether automatic posting of reviews is enabled or disabled.""" +        if await self.autoreview_enabled(): +            await ctx.send("Autoreview is currently enabled") +        else: +            await ctx.send("Autoreview is currently disabled") +      @nomination_group.command(name='watched', aliases=('all', 'list'), root_aliases=("nominees",))      @has_any_role(*MODERATION_ROLES)      async def watched_command( @@ -117,7 +178,7 @@ class TalentPool(WatchChannel, Cog, name="Talentpool"):      @nomination_group.command(name='forcewatch', aliases=('fw', 'forceadd', 'fa'), root_aliases=("forcenominate",))      @has_any_role(*MODERATION_ROLES) -    async def force_watch_command(self, ctx: Context, user: FetchedMember, *, reason: str = '') -> None: +    async def force_watch_command(self, ctx: Context, user: MemberOrUser, *, reason: str = '') -> None:          """          Adds the given `user` to the talent pool, from any channel. @@ -127,7 +188,7 @@ class TalentPool(WatchChannel, Cog, name="Talentpool"):      @nomination_group.command(name='watch', aliases=('w', 'add', 'a'), root_aliases=("nominate",))      @has_any_role(*STAFF_ROLES) -    async def watch_command(self, ctx: Context, user: FetchedMember, *, reason: str = '') -> None: +    async def watch_command(self, ctx: Context, user: MemberOrUser, *, reason: str = '') -> None:          """          Adds the given `user` to the talent pool. @@ -146,7 +207,7 @@ class TalentPool(WatchChannel, Cog, name="Talentpool"):          await self._watch_user(ctx, user, reason) -    async def _watch_user(self, ctx: Context, user: FetchedMember, reason: str) -> None: +    async def _watch_user(self, ctx: Context, user: MemberOrUser, reason: str) -> None:          """Adds the given user to the talent pool."""          if user.bot:              await ctx.send(f":x: I'm sorry {ctx.author}, I'm afraid I can't do that. I only watch humans.") @@ -190,7 +251,7 @@ class TalentPool(WatchChannel, Cog, name="Talentpool"):          self.watched_users[user.id] = response_data -        if user.id not in self.reviewer: +        if await self.autoreview_enabled() and user.id not in self.reviewer:              self.reviewer.schedule_review(user.id)          history = await self.bot.api_client.get( @@ -210,7 +271,7 @@ class TalentPool(WatchChannel, Cog, name="Talentpool"):      @nomination_group.command(name='history', aliases=('info', 'search'))      @has_any_role(*MODERATION_ROLES) -    async def history_command(self, ctx: Context, user: FetchedMember) -> None: +    async def history_command(self, ctx: Context, user: MemberOrUser) -> None:          """Shows the specified user's nomination history."""          result = await self.bot.api_client.get(              self.api_endpoint, @@ -239,7 +300,7 @@ class TalentPool(WatchChannel, Cog, name="Talentpool"):      @nomination_group.command(name='unwatch', aliases=('end', ), root_aliases=("unnominate",))      @has_any_role(*MODERATION_ROLES) -    async def unwatch_command(self, ctx: Context, user: FetchedMember, *, reason: str) -> None: +    async def unwatch_command(self, ctx: Context, user: MemberOrUser, *, reason: str) -> None:          """          Ends the active nomination of the specified user with the given reason. @@ -262,7 +323,7 @@ class TalentPool(WatchChannel, Cog, name="Talentpool"):      @nomination_edit_group.command(name='reason')      @has_any_role(*MODERATION_ROLES) -    async def edit_reason_command(self, ctx: Context, nomination_id: int, actor: FetchedMember, *, reason: str) -> None: +    async def edit_reason_command(self, ctx: Context, nomination_id: int, actor: MemberOrUser, *, reason: str) -> None:          """Edits the reason of a specific nominator in a specific active nomination."""          if len(reason) > REASON_MAX_CHARS:              await ctx.send(f":x: Maxiumum allowed characters for the reason is {REASON_MAX_CHARS}.") @@ -356,7 +417,7 @@ class TalentPool(WatchChannel, Cog, name="Talentpool"):          await ctx.message.add_reaction(Emojis.check_mark)      @Cog.listener() -    async def on_member_ban(self, guild: Guild, user: Union[User, Member]) -> None: +    async def on_member_ban(self, guild: Guild, user: Union[MemberOrUser]) -> None:          """Remove `user` from the talent pool after they are banned."""          await self.unwatch(user.id, "User was banned.") @@ -403,7 +464,8 @@ class TalentPool(WatchChannel, Cog, name="Talentpool"):          )          self._remove_user(user_id) -        self.reviewer.cancel(user_id) +        if await self.autoreview_enabled(): +            self.reviewer.cancel(user_id)          return True diff --git a/bot/exts/recruitment/talentpool/_review.py b/bot/exts/recruitment/talentpool/_review.py index 0cb786e4b..4d496a1f7 100644 --- a/bot/exts/recruitment/talentpool/_review.py +++ b/bot/exts/recruitment/talentpool/_review.py @@ -10,16 +10,15 @@ from datetime import datetime, timedelta  from typing import List, Optional, Union  from dateutil.parser import isoparse -from dateutil.relativedelta import relativedelta  from discord import Embed, Emoji, Member, Message, NoMoreItems, PartialMessage, TextChannel  from discord.ext.commands import Context  from bot.api import ResponseCodeError  from bot.bot import Bot -from bot.constants import Channels, Colours, Emojis, Guild, Roles +from bot.constants import Channels, Colours, Emojis, Guild  from bot.utils.messages import count_unique_users_reaction, pin_no_system_message  from bot.utils.scheduling import Scheduler -from bot.utils.time import get_time_delta, humanize_delta, time_since +from bot.utils.time import get_time_delta, time_since  if typing.TYPE_CHECKING:      from bot.exts.recruitment.talentpool._cog import TalentPool @@ -31,11 +30,15 @@ MAX_DAYS_IN_POOL = 30  # Maximum amount of characters allowed in a message  MAX_MESSAGE_SIZE = 2000 +# Maximum amount of characters allowed in an embed +MAX_EMBED_SIZE = 4000 -# Regex finding the user ID of a user mention -MENTION_RE = re.compile(r"<@!?(\d+?)>") -# Regex matching role pings -ROLE_MENTION_RE = re.compile(r"<@&\d+>") +# Regex for finding the first message of a nomination, and extracting the nominee. +# Historic nominations will have 2 role mentions at the start, new ones won't, optionally match for this. +NOMINATION_MESSAGE_REGEX = re.compile( +    r"(?:<@&\d+> <@&\d+>\n)*?<@!?(\d+?)> \(.+#\d{4}\) for Helper!\n\n\*\*Nominated by:\*\*", +    re.MULTILINE +)  class Reviewer: @@ -117,7 +120,7 @@ class Reviewer:                  f"I tried to review the user with ID `{user_id}`, but they don't appear to be on the server :pensive:"              ), None -        opening = f"<@&{Roles.mod_team}> <@&{Roles.admins}>\n{member.mention} ({member}) for Helper!" +        opening = f"{member.mention} ({member}) for Helper!"          current_nominations = "\n\n".join(              f"**<@{entry['actor']}>:** {entry['reason'] or '*no reason given*'}" @@ -141,14 +144,14 @@ class Reviewer:          """Archive this vote to #nomination-archive."""          message = await message.fetch() -        # We consider the first message in the nomination to contain the two role pings +        # We consider the first message in the nomination to contain the user ping, username#discrim, and fixed text          messages = [message] -        if not len(ROLE_MENTION_RE.findall(message.content)) >= 2: +        if not NOMINATION_MESSAGE_REGEX.search(message.content):              with contextlib.suppress(NoMoreItems):                  async for new_message in message.channel.history(before=message.created_at):                      messages.append(new_message) -                    if len(ROLE_MENTION_RE.findall(new_message.content)) >= 2: +                    if NOMINATION_MESSAGE_REGEX.search(new_message.content):                          break          log.debug(f"Found {len(messages)} messages: {', '.join(str(m.id) for m in messages)}") @@ -160,7 +163,7 @@ class Reviewer:          content = "".join(parts)          # We assume that the first user mentioned is the user that we are voting on -        user_id = int(MENTION_RE.search(content).group(1)) +        user_id = int(NOMINATION_MESSAGE_REGEX.search(content).group(1))          # Get reaction counts          reviewed = await count_unique_users_reaction( @@ -199,7 +202,7 @@ class Reviewer:          channel = self.bot.get_channel(Channels.nomination_archive)          for number, part in enumerate( -                textwrap.wrap(embed_content, width=MAX_MESSAGE_SIZE, replace_whitespace=False, placeholder="") +                textwrap.wrap(embed_content, width=MAX_EMBED_SIZE, replace_whitespace=False, placeholder="")          ):              await channel.send(embed=Embed(                  title=embed_title if number == 0 else None, @@ -253,9 +256,9 @@ class Reviewer:                  last_channel = user_activity["top_channel_activity"][-1]                  channels += f", and {last_channel[1]} in {last_channel[0]}" -        time_on_server = humanize_delta(relativedelta(datetime.utcnow(), member.joined_at), max_units=2) +        joined_at_formatted = time_since(member.joined_at)          review = ( -            f"{member.name} has been on the server for **{time_on_server}**" +            f"{member.name} joined the server **{joined_at_formatted}**"              f" and has **{messages} messages**{channels}."          ) @@ -345,7 +348,7 @@ class Reviewer:          nomination_times = f"{num_entries} times" if num_entries > 1 else "once"          rejection_times = f"{len(history)} times" if len(history) > 1 else "once" -        end_time = time_since(isoparse(history[0]['ended_at']).replace(tzinfo=None), max_units=2) +        end_time = time_since(isoparse(history[0]['ended_at']).replace(tzinfo=None))          review = (              f"They were nominated **{nomination_times}** before" diff --git a/bot/exts/utils/jams.py b/bot/exts/utils/jams.py deleted file mode 100644 index 87ae847f6..000000000 --- a/bot/exts/utils/jams.py +++ /dev/null @@ -1,176 +0,0 @@ -import csv -import logging -import typing as t -from collections import defaultdict - -import discord -from discord.ext import commands - -from bot.bot import Bot -from bot.constants import Categories, Channels, Emojis, Roles - -log = logging.getLogger(__name__) - -MAX_CHANNELS = 50 -CATEGORY_NAME = "Code Jam" -TEAM_LEADERS_COLOUR = 0x11806a - - -class CodeJams(commands.Cog): -    """Manages the code-jam related parts of our server.""" - -    def __init__(self, bot: Bot): -        self.bot = bot - -    @commands.group() -    @commands.has_any_role(Roles.admins) -    async def codejam(self, ctx: commands.Context) -> None: -        """A Group of commands for managing Code Jams.""" -        if ctx.invoked_subcommand is None: -            await ctx.send_help(ctx.command) - -    @codejam.command() -    async def create(self, ctx: commands.Context, csv_file: t.Optional[str]) -> None: -        """ -        Create code-jam teams from a CSV file or a link to one, specifying the team names, leaders and members. - -        The CSV file must have 3 columns: 'Team Name', 'Team Member Discord ID', and 'Team Leader'. - -        This will create the text channels for the teams, and give the team leaders their roles. -        """ -        async with ctx.typing(): -            if csv_file: -                async with self.bot.http_session.get(csv_file) as response: -                    if response.status != 200: -                        await ctx.send(f"Got a bad response from the URL: {response.status}") -                        return - -                    csv_file = await response.text() - -            elif ctx.message.attachments: -                csv_file = (await ctx.message.attachments[0].read()).decode("utf8") -            else: -                raise commands.BadArgument("You must include either a CSV file or a link to one.") - -            teams = defaultdict(list) -            reader = csv.DictReader(csv_file.splitlines()) - -            for row in reader: -                member = ctx.guild.get_member(int(row["Team Member Discord ID"])) - -                if member is None: -                    log.trace(f"Got an invalid member ID: {row['Team Member Discord ID']}") -                    continue - -                teams[row["Team Name"]].append((member, row["Team Leader"].upper() == "Y")) - -            team_leaders = await ctx.guild.create_role(name="Code Jam Team Leaders", colour=TEAM_LEADERS_COLOUR) - -            for team_name, members in teams.items(): -                await self.create_team_channel(ctx.guild, team_name, members, team_leaders) - -            await self.create_team_leader_channel(ctx.guild, team_leaders) -            await ctx.send(f"{Emojis.check_mark} Created Code Jam with {len(teams)} teams.") - -    async def get_category(self, guild: discord.Guild) -> discord.CategoryChannel: -        """ -        Return a code jam category. - -        If all categories are full or none exist, create a new category. -        """ -        for category in guild.categories: -            if category.name == CATEGORY_NAME and len(category.channels) < MAX_CHANNELS: -                return category - -        return await self.create_category(guild) - -    async def create_category(self, guild: discord.Guild) -> discord.CategoryChannel: -        """Create a new code jam category and return it.""" -        log.info("Creating a new code jam category.") - -        category_overwrites = { -            guild.default_role: discord.PermissionOverwrite(read_messages=False), -            guild.me: discord.PermissionOverwrite(read_messages=True) -        } - -        category = await guild.create_category_channel( -            CATEGORY_NAME, -            overwrites=category_overwrites, -            reason="It's code jam time!" -        ) - -        await self.send_status_update( -            guild, f"Created a new category with the ID {category.id} for this Code Jam's team channels." -        ) - -        return category - -    @staticmethod -    def get_overwrites( -        members: list[tuple[discord.Member, bool]], -        guild: discord.Guild, -    ) -> dict[t.Union[discord.Member, discord.Role], discord.PermissionOverwrite]: -        """Get code jam team channels permission overwrites.""" -        team_channel_overwrites = { -            guild.default_role: discord.PermissionOverwrite(read_messages=False), -            guild.get_role(Roles.code_jam_event_team): discord.PermissionOverwrite(read_messages=True) -        } - -        for member, _ in members: -            team_channel_overwrites[member] = discord.PermissionOverwrite( -                read_messages=True -            ) - -        return team_channel_overwrites - -    async def create_team_channel( -        self, -        guild: discord.Guild, -        team_name: str, -        members: list[tuple[discord.Member, bool]], -        team_leaders: discord.Role -    ) -> None: -        """Create the team's text channel.""" -        await self.add_team_leader_roles(members, team_leaders) - -        # Get permission overwrites and category -        team_channel_overwrites = self.get_overwrites(members, guild) -        code_jam_category = await self.get_category(guild) - -        # Create a text channel for the team -        await code_jam_category.create_text_channel( -            team_name, -            overwrites=team_channel_overwrites, -        ) - -    async def create_team_leader_channel(self, guild: discord.Guild, team_leaders: discord.Role) -> None: -        """Create the Team Leader Chat channel for the Code Jam team leaders.""" -        category: discord.CategoryChannel = guild.get_channel(Categories.summer_code_jam) - -        team_leaders_chat = await category.create_text_channel( -            name="team-leaders-chat", -            overwrites={ -                guild.default_role: discord.PermissionOverwrite(read_messages=False), -                team_leaders: discord.PermissionOverwrite(read_messages=True) -            } -        ) - -        await self.send_status_update(guild, f"Created {team_leaders_chat.mention} in the {category} category.") - -    async def send_status_update(self, guild: discord.Guild, message: str) -> None: -        """Inform the events lead with a status update when the command is ran.""" -        channel: discord.TextChannel = guild.get_channel(Channels.code_jam_planning) - -        await channel.send(f"<@&{Roles.events_lead}>\n\n{message}") - -    @staticmethod -    async def add_team_leader_roles(members: list[tuple[discord.Member, bool]], team_leaders: discord.Role) -> None: -        """Assign team leader role, the jammer role and their team role.""" -        for member, is_leader in members: -            if is_leader: -                await member.add_roles(team_leaders) - - -def setup(bot: Bot) -> None: -    """Load the CodeJams cog.""" -    bot.add_cog(CodeJams(bot)) diff --git a/bot/exts/utils/reminders.py b/bot/exts/utils/reminders.py index 6c21920a1..144f7b537 100644 --- a/bot/exts/utils/reminders.py +++ b/bot/exts/utils/reminders.py @@ -3,23 +3,22 @@ import logging  import random  import textwrap  import typing as t -from datetime import datetime, timedelta +from datetime import datetime  from operator import itemgetter  import discord  from dateutil.parser import isoparse -from dateutil.relativedelta import relativedelta  from discord.ext.commands import Cog, Context, Greedy, group  from bot.bot import Bot  from bot.constants import Guild, Icons, MODERATION_ROLES, POSITIVE_REPLIES, Roles, STAFF_ROLES -from bot.converters import Duration +from bot.converters import Duration, UserMentionOrID  from bot.pagination import LinePaginator  from bot.utils.checks import has_any_role_check, has_no_roles_check  from bot.utils.lock import lock_arg  from bot.utils.messages import send_denial  from bot.utils.scheduling import Scheduler -from bot.utils.time import humanize_delta +from bot.utils.time import TimestampFormats, discord_timestamp  log = logging.getLogger(__name__) @@ -28,6 +27,7 @@ WHITELISTED_CHANNELS = Guild.reminder_whitelist  MAXIMUM_REMINDERS = 5  Mentionable = t.Union[discord.Member, discord.Role] +ReminderMention = t.Union[UserMentionOrID, discord.Role]  class Reminders(Cog): @@ -62,8 +62,7 @@ class Reminders(Cog):              # If the reminder is already overdue ...              if remind_at < now: -                late = relativedelta(now, remind_at) -                await self.send_reminder(reminder, late) +                await self.send_reminder(reminder, remind_at)              else:                  self.schedule_reminder(reminder) @@ -86,8 +85,7 @@ class Reminders(Cog):      async def _send_confirmation(          ctx: Context,          on_success: str, -        reminder_id: t.Union[str, int], -        delivery_dt: t.Optional[datetime], +        reminder_id: t.Union[str, int]      ) -> None:          """Send an embed confirming the reminder change was made successfully."""          embed = discord.Embed( @@ -98,11 +96,6 @@ class Reminders(Cog):          footer_str = f"ID: {reminder_id}" -        if delivery_dt: -            # Reminder deletion will have a `None` `delivery_dt` -            footer_str += ', Due' -            embed.timestamp = delivery_dt -          embed.set_footer(text=footer_str)          await ctx.send(embed=embed) @@ -174,51 +167,59 @@ class Reminders(Cog):          self.schedule_reminder(reminder)      @lock_arg(LOCK_NAMESPACE, "reminder", itemgetter("id"), raise_error=True) -    async def send_reminder(self, reminder: dict, late: relativedelta = None) -> None: +    async def send_reminder(self, reminder: dict, expected_time: datetime = None) -> None:          """Send the reminder."""          is_valid, user, channel = self.ensure_valid_reminder(reminder)          if not is_valid:              # No need to cancel the task too; it'll simply be done once this coroutine returns.              return -          embed = discord.Embed() -        embed.colour = discord.Colour.blurple() -        embed.set_author( -            icon_url=Icons.remind_blurple, -            name="It has arrived!" -        ) - -        embed.description = f"Here's your reminder: `{reminder['content']}`." - -        if reminder.get("jump_url"):  # keep backward compatibility -            embed.description += f"\n[Jump back to when you created the reminder]({reminder['jump_url']})" - -        if late: +        if expected_time:              embed.colour = discord.Colour.red()              embed.set_author(                  icon_url=Icons.remind_red, -                name=f"Sorry it arrived {humanize_delta(late, max_units=2)} late!" +                name="Sorry, your reminder should have arrived earlier!"              ) +        else: +            embed.colour = discord.Colour.blurple() +            embed.set_author( +                icon_url=Icons.remind_blurple, +                name="It has arrived!" +            ) + +        # Let's not use a codeblock to keep emojis and mentions working. Embeds are safe anyway. +        embed.description = f"Here's your reminder: {reminder['content']}" +        # Here the jump URL is in the format of base_url/guild_id/channel_id/message_id          additional_mentions = ' '.join(              mentionable.mention for mentionable in self.get_mentionables(reminder["mentions"])          ) -        await channel.send(content=f"{user.mention} {additional_mentions}", embed=embed) +        jump_url = reminder.get("jump_url") +        embed.description += f"\n[Jump back to when you created the reminder]({jump_url})" +        partial_message = channel.get_partial_message(int(jump_url.split("/")[-1])) +        try: +            await partial_message.reply(content=f"{additional_mentions}", embed=embed) +        except discord.HTTPException as e: +            log.info( +                f"There was an error when trying to reply to a reminder invocation message, {e}, " +                "fall back to using jump_url" +            ) +            await channel.send(content=f"{user.mention} {additional_mentions}", embed=embed)          log.debug(f"Deleting reminder #{reminder['id']} (the user has been reminded).")          await self.bot.api_client.delete(f"bot/reminders/{reminder['id']}")      @group(name="remind", aliases=("reminder", "reminders", "remindme"), invoke_without_command=True)      async def remind_group( -        self, ctx: Context, mentions: Greedy[Mentionable], expiration: Duration, *, content: str +        self, ctx: Context, mentions: Greedy[ReminderMention], expiration: Duration, *, content: str      ) -> None:          """Commands for managing your reminders."""          await self.new_reminder(ctx, mentions=mentions, expiration=expiration, content=content)      @remind_group.command(name="new", aliases=("add", "create"))      async def new_reminder( -        self, ctx: Context, mentions: Greedy[Mentionable], expiration: Duration, *, content: str +        self, ctx: Context, mentions: Greedy[ReminderMention], expiration: Duration, *, content: str      ) -> None:          """          Set yourself a simple reminder. @@ -270,9 +271,7 @@ class Reminders(Cog):              }          ) -        now = datetime.utcnow() - timedelta(seconds=1) -        humanized_delta = humanize_delta(relativedelta(expiration, now)) -        mention_string = f"Your reminder will arrive in {humanized_delta}" +        mention_string = f"Your reminder will arrive on {discord_timestamp(expiration, TimestampFormats.DAY_TIME)}"          if mentions:              mention_string += f" and will mention {len(mentions)} other(s)" @@ -282,8 +281,7 @@ class Reminders(Cog):          await self._send_confirmation(              ctx,              on_success=mention_string, -            reminder_id=reminder["id"], -            delivery_dt=expiration, +            reminder_id=reminder["id"]          )          self.schedule_reminder(reminder) @@ -297,8 +295,6 @@ class Reminders(Cog):              params={'author__id': str(ctx.author.id)}          ) -        now = datetime.utcnow() -          # Make a list of tuples so it can be sorted by time.          reminders = sorted(              ( @@ -313,7 +309,7 @@ class Reminders(Cog):          for content, remind_at, id_, mentions in reminders:              # Parse and humanize the time, make it pretty :D              remind_datetime = isoparse(remind_at).replace(tzinfo=None) -            time = humanize_delta(relativedelta(remind_datetime, now)) +            time = discord_timestamp(remind_datetime, TimestampFormats.RELATIVE)              mentions = ", ".join(                  # Both Role and User objects have the `name` attribute @@ -322,7 +318,7 @@ class Reminders(Cog):              mention_string = f"\n**Mentions:** {mentions}" if mentions else ""              text = textwrap.dedent(f""" -            **Reminder #{id_}:** *expires in {time}* (ID: {id_}){mention_string} +            **Reminder #{id_}:** *expires {time}* (ID: {id_}){mention_string}              {content}              """).strip() @@ -368,7 +364,7 @@ class Reminders(Cog):          await self.edit_reminder(ctx, id_, {"content": content})      @edit_reminder_group.command(name="mentions", aliases=("pings",)) -    async def edit_reminder_mentions(self, ctx: Context, id_: int, mentions: Greedy[Mentionable]) -> None: +    async def edit_reminder_mentions(self, ctx: Context, id_: int, mentions: Greedy[ReminderMention]) -> None:          """Edit one of your reminder's mentions."""          # Remove duplicate mentions          mentions = set(mentions) @@ -388,15 +384,11 @@ class Reminders(Cog):              return          reminder = await self._edit_reminder(id_, payload) -        # Parse the reminder expiration back into a datetime -        expiration = isoparse(reminder["expiration"]).replace(tzinfo=None) -          # Send a confirmation message to the channel          await self._send_confirmation(              ctx,              on_success="That reminder has been edited successfully!",              reminder_id=id_, -            delivery_dt=expiration,          )          await self._reschedule_reminder(reminder) @@ -413,8 +405,7 @@ class Reminders(Cog):          await self._send_confirmation(              ctx,              on_success="That reminder has been deleted successfully!", -            reminder_id=id_, -            delivery_dt=None, +            reminder_id=id_          )      async def _can_modify(self, ctx: Context, reminder_id: t.Union[str, int]) -> bool: diff --git a/bot/exts/utils/utils.py b/bot/exts/utils/utils.py index 3b8564aee..98e43c32b 100644 --- a/bot/exts/utils/utils.py +++ b/bot/exts/utils/utils.py @@ -50,7 +50,7 @@ class Utils(Cog):          self.bot = bot      @command() -    @in_whitelist(channels=(Channels.bot_commands,), roles=STAFF_ROLES) +    @in_whitelist(channels=(Channels.bot_commands, Channels.discord_py), roles=STAFF_ROLES)      async def charinfo(self, ctx: Context, *, characters: str) -> None:          """Shows you information on up to 50 unicode characters."""          match = re.match(r"<(a?):(\w+):(\d+)>", characters) @@ -175,7 +175,7 @@ class Utils(Cog):          lines = []          for snowflake in snowflakes:              created_at = snowflake_time(snowflake) -            lines.append(f"**{snowflake}**\nCreated at {created_at} ({time_since(created_at, max_units=3)}).") +            lines.append(f"**{snowflake}**\nCreated at {created_at} ({time_since(created_at)}).")          await LinePaginator.paginate(              lines, diff --git a/bot/pagination.py b/bot/pagination.py index fbab74021..26caa7db0 100644 --- a/bot/pagination.py +++ b/bot/pagination.py @@ -49,8 +49,8 @@ class LinePaginator(Paginator):          self,          prefix: str = '```',          suffix: str = '```', -        max_size: int = 2000, -        scale_to_size: int = 2000, +        max_size: int = 4000, +        scale_to_size: int = 4000,          max_lines: t.Optional[int] = None,          linesep: str = "\n"      ) -> None: @@ -59,10 +59,10 @@ class LinePaginator(Paginator):          It overrides in order to allow us to configure the maximum number of lines per page.          """ -        # Embeds that exceed 2048 characters will result in an HTTPException -        # (Discord API limit), so we've set a limit of 2000 -        if max_size > 2000: -            raise ValueError(f"max_size must be <= 2,000 characters. ({max_size} > 2000)") +        # Embeds that exceed 4096 characters will result in an HTTPException +        # (Discord API limit), so we've set a limit of 4000 +        if max_size > 4000: +            raise ValueError(f"max_size must be <= 4,000 characters. ({max_size} > 4000)")          super().__init__(              prefix, @@ -74,8 +74,8 @@ class LinePaginator(Paginator):          if scale_to_size < max_size:              raise ValueError(f"scale_to_size must be >= max_size. ({scale_to_size} < {max_size})") -        if scale_to_size > 2000: -            raise ValueError(f"scale_to_size must be <= 2,000 characters. ({scale_to_size} > 2000)") +        if scale_to_size > 4000: +            raise ValueError(f"scale_to_size must be <= 4,000 characters. ({scale_to_size} > 4000)")          self.scale_to_size = scale_to_size - len(suffix)          self.max_lines = max_lines @@ -197,7 +197,7 @@ class LinePaginator(Paginator):          suffix: str = "",          max_lines: t.Optional[int] = None,          max_size: int = 500, -        scale_to_size: int = 2000, +        scale_to_size: int = 4000,          empty: bool = True,          restrict_to_user: User = None,          timeout: int = 300, diff --git a/bot/resources/tags/blocking.md b/bot/resources/tags/blocking.md index 31d91294c..5554d7eba 100644 --- a/bot/resources/tags/blocking.md +++ b/bot/resources/tags/blocking.md @@ -1,9 +1,7 @@  **Why do we need asynchronous programming?** -  Imagine that you're coding a Discord bot and every time somebody uses a command, you need to get some information from a database. But there's a catch: the database servers are acting up today and take a whole 10 seconds to respond. If you do **not** use asynchronous methods, your whole bot will stop running until it gets a response from the database. How do you fix this? Asynchronous programming.  **What is asynchronous programming?** -  An asynchronous program utilises the `async` and `await` keywords. An asynchronous program pauses what it's doing and does something else whilst it waits for some third-party service to complete whatever it's supposed to do. Any code within an `async` context manager or function marked with the `await` keyword indicates to Python, that whilst this operation is being completed, it can do something else. For example:  ```py @@ -14,13 +12,10 @@ import discord  async def ping(ctx):      await ctx.send("Pong!")  ``` -  **What does the term "blocking" mean?** -  A blocking operation is wherever you do something without `await`ing it. This tells Python that this step must be completed before it can do anything else. Common examples of blocking operations, as simple as they may seem, include: outputting text, adding two numbers and appending an item onto a list. Most common Python libraries have an asynchronous version available to use in asynchronous contexts.  **`async` libraries** -  The standard async library - `asyncio`  Asynchronous web requests - `aiohttp`  Talking to PostgreSQL asynchronously - `asyncpg` diff --git a/bot/resources/tags/docstring.md b/bot/resources/tags/docstring.md new file mode 100644 index 000000000..20043131e --- /dev/null +++ b/bot/resources/tags/docstring.md @@ -0,0 +1,18 @@ +A [`docstring`](https://docs.python.org/3/glossary.html#term-docstring) is a string - always using triple quotes - that's placed at the top of files, classes and functions. A docstring should contain a clear explanation of what it's describing. You can also include descriptions of the subject's parameter(s) and what it returns, as shown below: +```py +def greet(name: str, age: int) -> str: +    """ +    Return a string that greets the given person, using their name and age. + +    :param name: The name of the person to greet. +    :param age: The age of the person to greet. + +    :return: The greeting. +    """ +    return f"Hello {name}, you are {age} years old!" +``` +You can get the docstring by using the [`inspect.getdoc`](https://docs.python.org/3/library/inspect.html#inspect.getdoc) function, from the built-in [`inspect`](https://docs.python.org/3/library/inspect.html) module, or by accessing the `.__doc__` attribute. `inspect.getdoc` is often preferred, as it clears indents from the docstring. + +For the last example, you can print it by doing this: `print(inspect.getdoc(greet))`. + +For more details about what a docstring is and its usage, check out this guide by [Real Python](https://realpython.com/documenting-python-code/#docstrings-background), or the [official docstring specification](https://www.python.org/dev/peps/pep-0257/#what-is-a-docstring). diff --git a/bot/resources/tags/modmail.md b/bot/resources/tags/modmail.md index 412468174..8ac19c8a7 100644 --- a/bot/resources/tags/modmail.md +++ b/bot/resources/tags/modmail.md @@ -6,4 +6,4 @@ It supports attachments, codeblocks, and reactions. As communication happens ove  **To use it, simply send a direct message to the bot.** -Should there be an urgent and immediate need for a moderator or admin to look at a channel, feel free to ping the <@&831776746206265384> or <@&267628507062992896> role instead. +Should there be an urgent and immediate need for a moderator to look at a channel, feel free to ping the <@&831776746206265384> role instead. diff --git a/bot/resources/tags/venv.md b/bot/resources/tags/venv.md new file mode 100644 index 000000000..a4fc62151 --- /dev/null +++ b/bot/resources/tags/venv.md @@ -0,0 +1,20 @@ +**Virtual Environments** + +Virtual environments are isolated Python environments, which make it easier to keep your system clean and manage dependencies. By default, when activated, only libraries and scripts installed in the virtual environment are accessible, preventing cross-project dependency conflicts, and allowing easy isolation of requirements. + +To create a new virtual environment, you can use the standard library `venv` module: `python3 -m venv .venv` (replace `python3` with `python` or `py` on Windows) + +Then, to activate the new virtual environment: + +**Windows** (PowerShell): `.venv\Scripts\Activate.ps1` +or (Command Prompt): `.venv\Scripts\activate.bat` +**MacOS / Linux** (Bash): `source .venv/bin/activate` + +Packages can then be installed to the virtual environment using `pip`, as normal. + +For more information, take a read of the [documentation](https://docs.python.org/3/library/venv.html). If you run code through your editor, check its documentation on how to make it use your virtual environment. For example, see the [VSCode](https://code.visualstudio.com/docs/python/environments#_select-and-activate-an-environment) or [PyCharm](https://www.jetbrains.com/help/pycharm/creating-virtual-environment.html) docs. + +Tools such as [poetry](https://python-poetry.org/docs/basic-usage/) and [pipenv](https://pipenv.pypa.io/en/latest/) can manage the creation of virtual environments as well as project dependencies, making packaging and installing your project easier. + +**Note:** When using Windows PowerShell, you may need to change the [execution policy](https://docs.microsoft.com/en-us/powershell/module/microsoft.powershell.core/about/about_execution_policies) first. This is only required once: +`Set-ExecutionPolicy -ExecutionPolicy RemoteSigned -Scope CurrentUser` diff --git a/bot/utils/messages.py b/bot/utils/messages.py index d4a921161..abeb04021 100644 --- a/bot/utils/messages.py +++ b/bot/utils/messages.py @@ -1,5 +1,4 @@  import asyncio -import contextlib  import logging  import random  import re @@ -8,8 +7,6 @@ from io import BytesIO  from typing import Callable, List, Optional, Sequence, Union  import discord -from discord import Message, MessageType, Reaction, User -from discord.errors import HTTPException  from discord.ext.commands import Context  import bot @@ -54,7 +51,7 @@ def reaction_check(          log.trace(f"Removing reaction {reaction} by {user} on {reaction.message.id}: disallowed user.")          scheduling.create_task(              reaction.message.remove_reaction(reaction.emoji, user), -            suppressed_exceptions=(HTTPException,), +            suppressed_exceptions=(discord.HTTPException,),              name=f"remove_reaction-{reaction}-{reaction.message.id}-{user}"          )          return False @@ -69,7 +66,9 @@ async def wait_for_deletion(      allow_mods: bool = True  ) -> None:      """ -    Wait for up to `timeout` seconds for a reaction by any of the specified `user_ids` to delete the message. +    Wait for any of `user_ids` to react with one of the `deletion_emojis` within `timeout` seconds to delete `message`. + +    If `timeout` expires then reactions are cleared to indicate the option to delete has expired.      An `attach_emojis` bool may be specified to determine whether to attach the given      `deletion_emojis` to the message in the given `context`. @@ -95,9 +94,15 @@ async def wait_for_deletion(          allow_mods=allow_mods,      ) -    with contextlib.suppress(asyncio.TimeoutError): -        await bot.instance.wait_for('reaction_add', check=check, timeout=timeout) -        await message.delete() +    try: +        try: +            await bot.instance.wait_for('reaction_add', check=check, timeout=timeout) +        except asyncio.TimeoutError: +            await message.clear_reactions() +        else: +            await message.delete() +    except discord.NotFound: +        log.trace(f"wait_for_deletion: message {message.id} deleted prematurely.")  async def send_attachments( @@ -146,7 +151,7 @@ async def send_attachments(                  large.append(attachment)              else:                  log.info(f"{failure_msg} because it's too large.") -        except HTTPException as e: +        except discord.HTTPException as e:              if link_large and e.status == 413:                  large.append(attachment)              else: @@ -167,8 +172,8 @@ async def send_attachments(  async def count_unique_users_reaction(      message: discord.Message, -    reaction_predicate: Callable[[Reaction], bool] = lambda _: True, -    user_predicate: Callable[[User], bool] = lambda _: True, +    reaction_predicate: Callable[[discord.Reaction], bool] = lambda _: True, +    user_predicate: Callable[[discord.User], bool] = lambda _: True,      count_bots: bool = True  ) -> int:      """ @@ -188,7 +193,7 @@ async def count_unique_users_reaction(      return len(unique_users) -async def pin_no_system_message(message: Message) -> bool: +async def pin_no_system_message(message: discord.Message) -> bool:      """Pin the given message, wait a couple of seconds and try to delete the system message."""      await message.pin() @@ -196,7 +201,7 @@ async def pin_no_system_message(message: Message) -> bool:      await asyncio.sleep(2)      # Search for the system message in the last 10 messages      async for historical_message in message.channel.history(limit=10): -        if historical_message.type == MessageType.pins_add: +        if historical_message.type == discord.MessageType.pins_add:              await historical_message.delete()              return True diff --git a/bot/utils/time.py b/bot/utils/time.py index d55a0e532..8cf7d623b 100644 --- a/bot/utils/time.py +++ b/bot/utils/time.py @@ -1,12 +1,13 @@  import datetime  import re -from typing import Optional +from enum import Enum +from typing import Optional, Union  import dateutil.parser  from dateutil.relativedelta import relativedelta  RFC1123_FORMAT = "%a, %d %b %Y %H:%M:%S GMT" -INFRACTION_FORMAT = "%Y-%m-%d %H:%M" +DISCORD_TIMESTAMP_REGEX = re.compile(r"<t:(\d+):f>")  _DURATION_REGEX = re.compile(      r"((?P<years>\d+?) ?(years|year|Y|y) ?)?" @@ -19,6 +20,25 @@ _DURATION_REGEX = re.compile(  ) +ValidTimestamp = Union[int, datetime.datetime, datetime.date, datetime.timedelta, relativedelta] + + +class TimestampFormats(Enum): +    """ +    Represents the different formats possible for Discord timestamps. + +    Examples are given in epoch time. +    """ + +    DATE_TIME = "f"  # January 1, 1970 1:00 AM +    DAY_TIME = "F"  # Thursday, January 1, 1970 1:00 AM +    DATE_SHORT = "d"  # 01/01/1970 +    DATE = "D"  # January 1, 1970 +    TIME = "t"  # 1:00 AM +    TIME_SECONDS = "T"  # 1:00:00 AM +    RELATIVE = "R"  # 52 years ago + +  def _stringify_time_unit(value: int, unit: str) -> str:      """      Returns a string to represent a value and time unit, ensuring that it uses the right plural form of the unit. @@ -40,6 +60,24 @@ def _stringify_time_unit(value: int, unit: str) -> str:          return f"{value} {unit}" +def discord_timestamp(timestamp: ValidTimestamp, format: TimestampFormats = TimestampFormats.DATE_TIME) -> str: +    """Create and format a Discord flavored markdown timestamp.""" +    if format not in TimestampFormats: +        raise ValueError(f"Format can only be one of {', '.join(TimestampFormats.args)}, not {format}.") + +    # Convert each possible timestamp class to an integer. +    if isinstance(timestamp, datetime.datetime): +        timestamp = (timestamp.replace(tzinfo=None) - datetime.datetime.utcfromtimestamp(0)).total_seconds() +    elif isinstance(timestamp, datetime.date): +        timestamp = (timestamp - datetime.date.fromtimestamp(0)).total_seconds() +    elif isinstance(timestamp, datetime.timedelta): +        timestamp = timestamp.total_seconds() +    elif isinstance(timestamp, relativedelta): +        timestamp = timestamp.seconds + +    return f"<t:{int(timestamp)}:{format.value}>" + +  def humanize_delta(delta: relativedelta, precision: str = "seconds", max_units: int = 6) -> str:      """      Returns a human-readable version of the relativedelta. @@ -87,7 +125,7 @@ def humanize_delta(delta: relativedelta, precision: str = "seconds", max_units:  def get_time_delta(time_string: str) -> str:      """Returns the time in human-readable time delta format."""      date_time = dateutil.parser.isoparse(time_string).replace(tzinfo=None) -    time_delta = time_since(date_time, precision="minutes", max_units=1) +    time_delta = time_since(date_time)      return time_delta @@ -123,19 +161,9 @@ def relativedelta_to_timedelta(delta: relativedelta) -> datetime.timedelta:      return utcnow + delta - utcnow -def time_since(past_datetime: datetime.datetime, precision: str = "seconds", max_units: int = 6) -> str: -    """ -    Takes a datetime and returns a human-readable string that describes how long ago that datetime was. - -    precision specifies the smallest unit of time to include (e.g. "seconds", "minutes"). -    max_units specifies the maximum number of units of time to include (e.g. 1 may include days but not hours). -    """ -    now = datetime.datetime.utcnow() -    delta = abs(relativedelta(now, past_datetime)) - -    humanized = humanize_delta(delta, precision, max_units) - -    return f"{humanized} ago" +def time_since(past_datetime: datetime.datetime) -> str: +    """Takes a datetime and returns a discord timestamp that describes how long ago that datetime was.""" +    return discord_timestamp(past_datetime, TimestampFormats.RELATIVE)  def parse_rfc1123(stamp: str) -> datetime.datetime: @@ -144,8 +172,8 @@ def parse_rfc1123(stamp: str) -> datetime.datetime:  def format_infraction(timestamp: str) -> str: -    """Format an infraction timestamp to a more readable ISO 8601 format.""" -    return dateutil.parser.isoparse(timestamp).strftime(INFRACTION_FORMAT) +    """Format an infraction timestamp to a discord timestamp.""" +    return discord_timestamp(dateutil.parser.isoparse(timestamp))  def format_infraction_with_duration( @@ -155,11 +183,7 @@ def format_infraction_with_duration(      absolute: bool = True  ) -> Optional[str]:      """ -    Return `date_to` formatted as a readable ISO-8601 with the humanized duration since `date_from`. - -    `date_from` must be an ISO-8601 formatted timestamp. The duration is calculated as from -    `date_from` until `date_to` with a precision of seconds. If `date_from` is unspecified, the -    current time is used. +    Return `date_to` formatted as a discord timestamp with the timestamp duration since `date_from`.      `max_units` specifies the maximum number of units of time to include in the duration. For      example, a value of 1 may include days but not hours. @@ -186,25 +210,22 @@ def format_infraction_with_duration(  def until_expiration( -    expiry: Optional[str], -    now: Optional[datetime.datetime] = None, -    max_units: int = 2 +    expiry: Optional[str]  ) -> Optional[str]:      """ -    Get the remaining time until infraction's expiration, in a human-readable version of the relativedelta. +    Get the remaining time until infraction's expiration, in a discord timestamp.      Returns a human-readable version of the remaining duration between datetime.utcnow() and an expiry. -    Unlike `humanize_delta`, this function will force the `precision` to be `seconds` by not passing it. -    `max_units` specifies the maximum number of units of time to include (e.g. 1 may include days but not hours). -    By default, max_units is 2. +    Similar to time_since, except that this function doesn't error on a null input +    and return null if the expiry is in the paste      """      if not expiry:          return None -    now = now or datetime.datetime.utcnow() +    now = datetime.datetime.utcnow()      since = dateutil.parser.isoparse(expiry).replace(tzinfo=None, microsecond=0)      if since < now:          return None -    return humanize_delta(relativedelta(since, now), max_units=max_units) +    return discord_timestamp(since, TimestampFormats.RELATIVE) diff --git a/config-default.yml b/config-default.yml index 811640034..79828dd77 100644 --- a/config-default.yml +++ b/config-default.yml @@ -260,7 +260,7 @@ guild:          contributors:                           295488872404484098          help_cooldown:                          699189276025421825          muted:              &MUTED_ROLE         277914926603829249 -        partners:                               323426753857191936 +        partners:           &PY_PARTNER_ROLE    323426753857191936          python_community:   &PY_COMMUNITY_ROLE  458226413825294336          sprinters:          &SPRINTERS          758422482289426471          voice_verified:                         764802720779337729 @@ -342,6 +342,7 @@ filter:          - *OWNERS_ROLE          - *PY_COMMUNITY_ROLE          - *SPRINTERS +        - *PY_PARTNER_ROLE  keys: @@ -431,14 +432,12 @@ anti_spam:              max: 3 -  metabase: -    username: !ENV "METABASE_USERNAME" -    password: !ENV "METABASE_PASSWORD" -    url: "http://metabase.default.svc.cluster.local/api" +    username: !ENV      "METABASE_USERNAME" +    password: !ENV      "METABASE_PASSWORD" +    base_url:           "http://metabase.default.svc.cluster.local"      # 14 days, see https://www.metabase.com/docs/latest/operations-guide/environment-variables.html#max_session_age -    max_session_age: 20160 - +    max_session_age:    20160  big_brother: diff --git a/poetry.lock b/poetry.lock index dac277ed8..a4ce5d1a9 100644 --- a/poetry.lock +++ b/poetry.lock @@ -71,14 +71,6 @@ yarl = "*"  develop = ["aiomisc (>=11.0,<12.0)", "async-generator", "coverage (!=4.3)", "coveralls", "pylava", "pytest", "pytest-cov", "tox (>=2.4)"]  [[package]] -name = "appdirs" -version = "1.4.4" -description = "A small Python module for determining appropriate platform-specific dirs, e.g. a \"user data dir\"." -category = "dev" -optional = false -python-versions = "*" - -[[package]]  name = "arrow"  version = "1.0.3"  description = "Better dates & times for Python" @@ -135,6 +127,18 @@ tests = ["coverage[toml] (>=5.0.2)", "hypothesis", "pympler", "pytest (>=4.3.0)"  tests_no_zope = ["coverage[toml] (>=5.0.2)", "hypothesis", "pympler", "pytest (>=4.3.0)", "six", "mypy", "pytest-mypy-plugins"]  [[package]] +name = "backports.entry-points-selectable" +version = "1.1.0" +description = "Compatibility shim providing selectable entry points for older implementations" +category = "dev" +optional = false +python-versions = ">=2.7" + +[package.extras] +docs = ["sphinx", "jaraco.packaging (>=8.2)", "rst.linker (>=1.9)"] +testing = ["pytest (>=4.6)", "pytest-flake8", "pytest-cov", "pytest-black (>=0.3.7)", "pytest-mypy", "pytest-checkdocs (>=2.4)", "pytest-enabler (>=1.0.1)"] + +[[package]]  name = "beautifulsoup4"  version = "4.9.3"  description = "Screen-scraping library" @@ -159,7 +163,7 @@ python-versions = "*"  [[package]]  name = "cffi" -version = "1.14.5" +version = "1.14.6"  description = "Foreign Function Interface for Python calling C code."  category = "main"  optional = false @@ -185,6 +189,17 @@ optional = false  python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*"  [[package]] +name = "charset-normalizer" +version = "2.0.4" +description = "The Real First Universal Charset Detector. Open, modern and actively maintained alternative to Chardet." +category = "dev" +optional = false +python-versions = ">=3.5.0" + +[package.extras] +unicode_backport = ["unicodedata2"] + +[[package]]  name = "colorama"  version = "0.4.4"  description = "Cross-platform colored terminal text." @@ -463,7 +478,7 @@ pyreadline = {version = "*", markers = "sys_platform == \"win32\""}  [[package]]  name = "identify" -version = "2.2.10" +version = "2.2.11"  description = "File identification library for Python"  category = "dev"  optional = false @@ -474,11 +489,11 @@ license = ["editdistance-s"]  [[package]]  name = "idna" -version = "2.10" +version = "3.2"  description = "Internationalized Domain Names in Applications (IDNA)"  category = "main"  optional = false -python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" +python-versions = ">=3.5"  [[package]]  name = "iniconfig" @@ -597,6 +612,18 @@ flake8 = ">=3.9.1"  flake8-polyfill = ">=1.0.2,<2"  [[package]] +name = "platformdirs" +version = "2.2.0" +description = "A small Python module for determining appropriate platform-specific dirs, e.g. a \"user data dir\"." +category = "dev" +optional = false +python-versions = ">=3.6" + +[package.extras] +docs = ["Sphinx (>=4)", "furo (>=2021.7.5b38)", "proselint (>=0.10.2)", "sphinx-autodoc-typehints (>=1.12)"] +test = ["appdirs (==1.4.4)", "pytest (>=6)", "pytest-cov (>=2.7)", "pytest-mock (>=3.6)"] + +[[package]]  name = "pluggy"  version = "0.13.1"  description = "plugin and hook calling mechanisms for python" @@ -779,7 +806,7 @@ testing = ["filelock"]  [[package]]  name = "python-dateutil" -version = "2.8.1" +version = "2.8.2"  description = "Extensions to the standard Python datetime module"  category = "main"  optional = false @@ -851,25 +878,25 @@ python-versions = "*"  [[package]]  name = "requests" -version = "2.25.1" +version = "2.26.0"  description = "Python HTTP for Humans."  category = "dev"  optional = false -python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*, !=3.5.*"  [package.dependencies]  certifi = ">=2017.4.17" -chardet = ">=3.0.2,<5" -idna = ">=2.5,<3" +charset-normalizer = {version = ">=2.0.0,<2.1.0", markers = "python_version >= \"3\""} +idna = {version = ">=2.5,<4", markers = "python_version >= \"3\""}  urllib3 = ">=1.21.1,<1.27"  [package.extras] -security = ["pyOpenSSL (>=0.14)", "cryptography (>=1.3.4)"]  socks = ["PySocks (>=1.5.6,!=1.5.7)", "win-inet-pton"] +use_chardet_on_py3 = ["chardet (>=3.0.2,<5)"]  [[package]]  name = "sentry-sdk" -version = "0.20.3" +version = "1.3.1"  description = "Python client for Sentry (https://sentry.io)"  category = "main"  optional = false @@ -888,6 +915,7 @@ chalice = ["chalice (>=1.16.0)"]  django = ["django (>=1.8)"]  falcon = ["falcon (>=1.4)"]  flask = ["flask (>=0.11)", "blinker (>=1.1)"] +httpx = ["httpx (>=0.16.0)"]  pure_eval = ["pure-eval", "executing", "asttokens"]  pyspark = ["pyspark (>=2.4.4)"]  rq = ["rq (>=0.6)"] @@ -987,21 +1015,22 @@ socks = ["PySocks (>=1.5.6,!=1.5.7,<2.0)"]  [[package]]  name = "virtualenv" -version = "20.4.7" +version = "20.7.0"  description = "Virtual Python Environment builder"  category = "dev"  optional = false -python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,>=2.7" +python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,>=2.7"  [package.dependencies] -appdirs = ">=1.4.3,<2" +"backports.entry-points-selectable" = ">=1.0.4"  distlib = ">=0.3.1,<1"  filelock = ">=3.0.0,<4" +platformdirs = ">=2,<3"  six = ">=1.9.0,<2"  [package.extras]  docs = ["proselint (>=0.10.2)", "sphinx (>=3)", "sphinx-argparse (>=0.2.5)", "sphinx-rtd-theme (>=0.4.3)", "towncrier (>=19.9.0rc1)"] -testing = ["coverage (>=4)", "coverage-enable-subprocess (>=1)", "flaky (>=3)", "pytest (>=4)", "pytest-env (>=0.6.2)", "pytest-freezegun (>=0.4.1)", "pytest-mock (>=2)", "pytest-randomly (>=1)", "pytest-timeout (>=1)", "packaging (>=20.0)", "xonsh (>=0.9.16)"] +testing = ["coverage (>=4)", "coverage-enable-subprocess (>=1)", "flaky (>=3)", "pytest (>=4)", "pytest-env (>=0.6.2)", "pytest-freezegun (>=0.4.1)", "pytest-mock (>=2)", "pytest-randomly (>=1)", "pytest-timeout (>=1)", "packaging (>=20.0)"]  [[package]]  name = "yarl" @@ -1018,7 +1047,7 @@ multidict = ">=4.0"  [metadata]  lock-version = "1.1"  python-versions = "3.9.*" -content-hash = "85160036e3b07c9d5d24a32302462591e82cc3bf3d5490b87550d9c26bc5648d" +content-hash = "f46fe1d2d9e0621e4e06d4c2ba5f6190ec4574ac6ca809abe8bf542a3b55204e"  [metadata.files]  aio-pika = [ @@ -1076,10 +1105,6 @@ aiormq = [      {file = "aiormq-3.3.1-py3-none-any.whl", hash = "sha256:e584dac13a242589aaf42470fd3006cb0dc5aed6506cbd20357c7ec8bbe4a89e"},      {file = "aiormq-3.3.1.tar.gz", hash = "sha256:8218dd9f7198d6e7935855468326bbacf0089f926c70baa8dd92944cb2496573"},  ] -appdirs = [ -    {file = "appdirs-1.4.4-py2.py3-none-any.whl", hash = "sha256:a841dacd6b99318a741b166adb07e19ee71a274450e68237b4650ca1055ab128"}, -    {file = "appdirs-1.4.4.tar.gz", hash = "sha256:7d5d0167b2b1ba821647616af46a749d1c653740dd0d2415100fe26e27afdf41"}, -]  arrow = [      {file = "arrow-1.0.3-py3-none-any.whl", hash = "sha256:3515630f11a15c61dcb4cdd245883270dd334c83f3e639824e65a4b79cc48543"},      {file = "arrow-1.0.3.tar.gz", hash = "sha256:399c9c8ae732270e1aa58ead835a79a40d7be8aa109c579898eb41029b5a231d"}, @@ -1100,6 +1125,10 @@ attrs = [      {file = "attrs-21.2.0-py2.py3-none-any.whl", hash = "sha256:149e90d6d8ac20db7a955ad60cf0e6881a3f20d37096140088356da6c716b0b1"},      {file = "attrs-21.2.0.tar.gz", hash = "sha256:ef6aaac3ca6cd92904cdd0d83f629a15f18053ec84e6432106f7a4d04ae4f5fb"},  ] +"backports.entry-points-selectable" = [ +    {file = "backports.entry_points_selectable-1.1.0-py2.py3-none-any.whl", hash = "sha256:a6d9a871cde5e15b4c4a53e3d43ba890cc6861ec1332c9c2428c92f977192acc"}, +    {file = "backports.entry_points_selectable-1.1.0.tar.gz", hash = "sha256:988468260ec1c196dab6ae1149260e2f5472c9110334e5d51adcb77867361f6a"}, +]  beautifulsoup4 = [      {file = "beautifulsoup4-4.9.3-py2-none-any.whl", hash = "sha256:4c98143716ef1cb40bf7f39a8e3eec8f8b009509e74904ba3a7b315431577e35"},      {file = "beautifulsoup4-4.9.3-py3-none-any.whl", hash = "sha256:fff47e031e34ec82bf17e00da8f592fe7de69aeea38be00523c04623c04fb666"}, @@ -1110,43 +1139,51 @@ certifi = [      {file = "certifi-2021.5.30.tar.gz", hash = "sha256:2bbf76fd432960138b3ef6dda3dde0544f27cbf8546c458e60baf371917ba9ee"},  ]  cffi = [ -    {file = "cffi-1.14.5-cp27-cp27m-macosx_10_9_x86_64.whl", hash = "sha256:bb89f306e5da99f4d922728ddcd6f7fcebb3241fc40edebcb7284d7514741991"}, -    {file = "cffi-1.14.5-cp27-cp27m-manylinux1_i686.whl", hash = "sha256:34eff4b97f3d982fb93e2831e6750127d1355a923ebaeeb565407b3d2f8d41a1"}, -    {file = "cffi-1.14.5-cp27-cp27m-manylinux1_x86_64.whl", hash = "sha256:99cd03ae7988a93dd00bcd9d0b75e1f6c426063d6f03d2f90b89e29b25b82dfa"}, -    {file = "cffi-1.14.5-cp27-cp27m-win32.whl", hash = "sha256:65fa59693c62cf06e45ddbb822165394a288edce9e276647f0046e1ec26920f3"}, -    {file = "cffi-1.14.5-cp27-cp27m-win_amd64.whl", hash = "sha256:51182f8927c5af975fece87b1b369f722c570fe169f9880764b1ee3bca8347b5"}, -    {file = "cffi-1.14.5-cp27-cp27mu-manylinux1_i686.whl", hash = "sha256:43e0b9d9e2c9e5d152946b9c5fe062c151614b262fda2e7b201204de0b99e482"}, -    {file = "cffi-1.14.5-cp27-cp27mu-manylinux1_x86_64.whl", hash = "sha256:cbde590d4faaa07c72bf979734738f328d239913ba3e043b1e98fe9a39f8b2b6"}, -    {file = "cffi-1.14.5-cp35-cp35m-macosx_10_9_x86_64.whl", hash = "sha256:5de7970188bb46b7bf9858eb6890aad302577a5f6f75091fd7cdd3ef13ef3045"}, -    {file = "cffi-1.14.5-cp35-cp35m-manylinux1_i686.whl", hash = "sha256:a465da611f6fa124963b91bf432d960a555563efe4ed1cc403ba5077b15370aa"}, -    {file = "cffi-1.14.5-cp35-cp35m-manylinux1_x86_64.whl", hash = "sha256:d42b11d692e11b6634f7613ad8df5d6d5f8875f5d48939520d351007b3c13406"}, -    {file = "cffi-1.14.5-cp35-cp35m-win32.whl", hash = "sha256:72d8d3ef52c208ee1c7b2e341f7d71c6fd3157138abf1a95166e6165dd5d4369"}, -    {file = "cffi-1.14.5-cp35-cp35m-win_amd64.whl", hash = "sha256:29314480e958fd8aab22e4a58b355b629c59bf5f2ac2492b61e3dc06d8c7a315"}, -    {file = "cffi-1.14.5-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:3d3dd4c9e559eb172ecf00a2a7517e97d1e96de2a5e610bd9b68cea3925b4892"}, -    {file = "cffi-1.14.5-cp36-cp36m-manylinux1_i686.whl", hash = "sha256:48e1c69bbacfc3d932221851b39d49e81567a4d4aac3b21258d9c24578280058"}, -    {file = "cffi-1.14.5-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:69e395c24fc60aad6bb4fa7e583698ea6cc684648e1ffb7fe85e3c1ca131a7d5"}, -    {file = "cffi-1.14.5-cp36-cp36m-manylinux2014_aarch64.whl", hash = "sha256:9e93e79c2551ff263400e1e4be085a1210e12073a31c2011dbbda14bda0c6132"}, -    {file = "cffi-1.14.5-cp36-cp36m-win32.whl", hash = "sha256:58e3f59d583d413809d60779492342801d6e82fefb89c86a38e040c16883be53"}, -    {file = "cffi-1.14.5-cp36-cp36m-win_amd64.whl", hash = "sha256:005a36f41773e148deac64b08f233873a4d0c18b053d37da83f6af4d9087b813"}, -    {file = "cffi-1.14.5-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:2894f2df484ff56d717bead0a5c2abb6b9d2bf26d6960c4604d5c48bbc30ee73"}, -    {file = "cffi-1.14.5-cp37-cp37m-manylinux1_i686.whl", hash = "sha256:0857f0ae312d855239a55c81ef453ee8fd24136eaba8e87a2eceba644c0d4c06"}, -    {file = "cffi-1.14.5-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:cd2868886d547469123fadc46eac7ea5253ea7fcb139f12e1dfc2bbd406427d1"}, -    {file = "cffi-1.14.5-cp37-cp37m-manylinux2014_aarch64.whl", hash = "sha256:35f27e6eb43380fa080dccf676dece30bef72e4a67617ffda586641cd4508d49"}, -    {file = "cffi-1.14.5-cp37-cp37m-win32.whl", hash = "sha256:9ff227395193126d82e60319a673a037d5de84633f11279e336f9c0f189ecc62"}, -    {file = "cffi-1.14.5-cp37-cp37m-win_amd64.whl", hash = "sha256:9cf8022fb8d07a97c178b02327b284521c7708d7c71a9c9c355c178ac4bbd3d4"}, -    {file = "cffi-1.14.5-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:8b198cec6c72df5289c05b05b8b0969819783f9418e0409865dac47288d2a053"}, -    {file = "cffi-1.14.5-cp38-cp38-manylinux1_i686.whl", hash = "sha256:ad17025d226ee5beec591b52800c11680fca3df50b8b29fe51d882576e039ee0"}, -    {file = "cffi-1.14.5-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:6c97d7350133666fbb5cf4abdc1178c812cb205dc6f41d174a7b0f18fb93337e"}, -    {file = "cffi-1.14.5-cp38-cp38-manylinux2014_aarch64.whl", hash = "sha256:8ae6299f6c68de06f136f1f9e69458eae58f1dacf10af5c17353eae03aa0d827"}, -    {file = "cffi-1.14.5-cp38-cp38-win32.whl", hash = "sha256:b85eb46a81787c50650f2392b9b4ef23e1f126313b9e0e9013b35c15e4288e2e"}, -    {file = "cffi-1.14.5-cp38-cp38-win_amd64.whl", hash = "sha256:1f436816fc868b098b0d63b8920de7d208c90a67212546d02f84fe78a9c26396"}, -    {file = "cffi-1.14.5-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:1071534bbbf8cbb31b498d5d9db0f274f2f7a865adca4ae429e147ba40f73dea"}, -    {file = "cffi-1.14.5-cp39-cp39-manylinux1_i686.whl", hash = "sha256:9de2e279153a443c656f2defd67769e6d1e4163952b3c622dcea5b08a6405322"}, -    {file = "cffi-1.14.5-cp39-cp39-manylinux1_x86_64.whl", hash = "sha256:6e4714cc64f474e4d6e37cfff31a814b509a35cb17de4fb1999907575684479c"}, -    {file = "cffi-1.14.5-cp39-cp39-manylinux2014_aarch64.whl", hash = "sha256:158d0d15119b4b7ff6b926536763dc0714313aa59e320ddf787502c70c4d4bee"}, -    {file = "cffi-1.14.5-cp39-cp39-win32.whl", hash = "sha256:afb29c1ba2e5a3736f1c301d9d0abe3ec8b86957d04ddfa9d7a6a42b9367e396"}, -    {file = "cffi-1.14.5-cp39-cp39-win_amd64.whl", hash = "sha256:f2d45f97ab6bb54753eab54fffe75aaf3de4ff2341c9daee1987ee1837636f1d"}, -    {file = "cffi-1.14.5.tar.gz", hash = "sha256:fd78e5fee591709f32ef6edb9a015b4aa1a5022598e36227500c8f4e02328d9c"}, +    {file = "cffi-1.14.6-cp27-cp27m-macosx_10_9_x86_64.whl", hash = "sha256:22b9c3c320171c108e903d61a3723b51e37aaa8c81255b5e7ce102775bd01e2c"}, +    {file = "cffi-1.14.6-cp27-cp27m-manylinux1_i686.whl", hash = "sha256:f0c5d1acbfca6ebdd6b1e3eded8d261affb6ddcf2186205518f1428b8569bb99"}, +    {file = "cffi-1.14.6-cp27-cp27m-manylinux1_x86_64.whl", hash = "sha256:99f27fefe34c37ba9875f224a8f36e31d744d8083e00f520f133cab79ad5e819"}, +    {file = "cffi-1.14.6-cp27-cp27m-win32.whl", hash = "sha256:55af55e32ae468e9946f741a5d51f9896da6b9bf0bbdd326843fec05c730eb20"}, +    {file = "cffi-1.14.6-cp27-cp27m-win_amd64.whl", hash = "sha256:7bcac9a2b4fdbed2c16fa5681356d7121ecabf041f18d97ed5b8e0dd38a80224"}, +    {file = "cffi-1.14.6-cp27-cp27mu-manylinux1_i686.whl", hash = "sha256:ed38b924ce794e505647f7c331b22a693bee1538fdf46b0222c4717b42f744e7"}, +    {file = "cffi-1.14.6-cp27-cp27mu-manylinux1_x86_64.whl", hash = "sha256:e22dcb48709fc51a7b58a927391b23ab37eb3737a98ac4338e2448bef8559b33"}, +    {file = "cffi-1.14.6-cp35-cp35m-macosx_10_9_x86_64.whl", hash = "sha256:aedb15f0a5a5949ecb129a82b72b19df97bbbca024081ed2ef88bd5c0a610534"}, +    {file = "cffi-1.14.6-cp35-cp35m-manylinux1_i686.whl", hash = "sha256:48916e459c54c4a70e52745639f1db524542140433599e13911b2f329834276a"}, +    {file = "cffi-1.14.6-cp35-cp35m-manylinux1_x86_64.whl", hash = "sha256:f627688813d0a4140153ff532537fbe4afea5a3dffce1f9deb7f91f848a832b5"}, +    {file = "cffi-1.14.6-cp35-cp35m-win32.whl", hash = "sha256:f0010c6f9d1a4011e429109fda55a225921e3206e7f62a0c22a35344bfd13cca"}, +    {file = "cffi-1.14.6-cp35-cp35m-win_amd64.whl", hash = "sha256:57e555a9feb4a8460415f1aac331a2dc833b1115284f7ded7278b54afc5bd218"}, +    {file = "cffi-1.14.6-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:e8c6a99be100371dbb046880e7a282152aa5d6127ae01783e37662ef73850d8f"}, +    {file = "cffi-1.14.6-cp36-cp36m-manylinux1_i686.whl", hash = "sha256:19ca0dbdeda3b2615421d54bef8985f72af6e0c47082a8d26122adac81a95872"}, +    {file = "cffi-1.14.6-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:d950695ae4381ecd856bcaf2b1e866720e4ab9a1498cba61c602e56630ca7195"}, +    {file = "cffi-1.14.6-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:e9dc245e3ac69c92ee4c167fbdd7428ec1956d4e754223124991ef29eb57a09d"}, +    {file = "cffi-1.14.6-cp36-cp36m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:a8661b2ce9694ca01c529bfa204dbb144b275a31685a075ce123f12331be790b"}, +    {file = "cffi-1.14.6-cp36-cp36m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:b315d709717a99f4b27b59b021e6207c64620790ca3e0bde636a6c7f14618abb"}, +    {file = "cffi-1.14.6-cp36-cp36m-win32.whl", hash = "sha256:80b06212075346b5546b0417b9f2bf467fea3bfe7352f781ffc05a8ab24ba14a"}, +    {file = "cffi-1.14.6-cp36-cp36m-win_amd64.whl", hash = "sha256:a9da7010cec5a12193d1af9872a00888f396aba3dc79186604a09ea3ee7c029e"}, +    {file = "cffi-1.14.6-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:4373612d59c404baeb7cbd788a18b2b2a8331abcc84c3ba40051fcd18b17a4d5"}, +    {file = "cffi-1.14.6-cp37-cp37m-manylinux1_i686.whl", hash = "sha256:f10afb1004f102c7868ebfe91c28f4a712227fe4cb24974350ace1f90e1febbf"}, +    {file = "cffi-1.14.6-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:fd4305f86f53dfd8cd3522269ed7fc34856a8ee3709a5e28b2836b2db9d4cd69"}, +    {file = "cffi-1.14.6-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6d6169cb3c6c2ad50db5b868db6491a790300ade1ed5d1da29289d73bbe40b56"}, +    {file = "cffi-1.14.6-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:5d4b68e216fc65e9fe4f524c177b54964af043dde734807586cf5435af84045c"}, +    {file = "cffi-1.14.6-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:33791e8a2dc2953f28b8d8d300dde42dd929ac28f974c4b4c6272cb2955cb762"}, +    {file = "cffi-1.14.6-cp37-cp37m-win32.whl", hash = "sha256:0c0591bee64e438883b0c92a7bed78f6290d40bf02e54c5bf0978eaf36061771"}, +    {file = "cffi-1.14.6-cp37-cp37m-win_amd64.whl", hash = "sha256:8eb687582ed7cd8c4bdbff3df6c0da443eb89c3c72e6e5dcdd9c81729712791a"}, +    {file = "cffi-1.14.6-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:ba6f2b3f452e150945d58f4badd92310449876c4c954836cfb1803bdd7b422f0"}, +    {file = "cffi-1.14.6-cp38-cp38-manylinux1_i686.whl", hash = "sha256:64fda793737bc4037521d4899be780534b9aea552eb673b9833b01f945904c2e"}, +    {file = "cffi-1.14.6-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:9f3e33c28cd39d1b655ed1ba7247133b6f7fc16fa16887b120c0c670e35ce346"}, +    {file = "cffi-1.14.6-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:26bb2549b72708c833f5abe62b756176022a7b9a7f689b571e74c8478ead51dc"}, +    {file = "cffi-1.14.6-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:eb687a11f0a7a1839719edd80f41e459cc5366857ecbed383ff376c4e3cc6afd"}, +    {file = "cffi-1.14.6-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:d2ad4d668a5c0645d281dcd17aff2be3212bc109b33814bbb15c4939f44181cc"}, +    {file = "cffi-1.14.6-cp38-cp38-win32.whl", hash = "sha256:487d63e1454627c8e47dd230025780e91869cfba4c753a74fda196a1f6ad6548"}, +    {file = "cffi-1.14.6-cp38-cp38-win_amd64.whl", hash = "sha256:c33d18eb6e6bc36f09d793c0dc58b0211fccc6ae5149b808da4a62660678b156"}, +    {file = "cffi-1.14.6-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:06c54a68935738d206570b20da5ef2b6b6d92b38ef3ec45c5422c0ebaf338d4d"}, +    {file = "cffi-1.14.6-cp39-cp39-manylinux1_i686.whl", hash = "sha256:f174135f5609428cc6e1b9090f9268f5c8935fddb1b25ccb8255a2d50de6789e"}, +    {file = "cffi-1.14.6-cp39-cp39-manylinux1_x86_64.whl", hash = "sha256:f3ebe6e73c319340830a9b2825d32eb6d8475c1dac020b4f0aa774ee3b898d1c"}, +    {file = "cffi-1.14.6-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3c8d896becff2fa653dc4438b54a5a25a971d1f4110b32bd3068db3722c80202"}, +    {file = "cffi-1.14.6-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:4922cd707b25e623b902c86188aca466d3620892db76c0bdd7b99a3d5e61d35f"}, +    {file = "cffi-1.14.6-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:c9e005e9bd57bc987764c32a1bee4364c44fdc11a3cc20a40b93b444984f2b87"}, +    {file = "cffi-1.14.6-cp39-cp39-win32.whl", hash = "sha256:eb9e2a346c5238a30a746893f23a9535e700f8192a68c07c0258e7ece6ff3728"}, +    {file = "cffi-1.14.6-cp39-cp39-win_amd64.whl", hash = "sha256:818014c754cd3dba7229c0f5884396264d51ffb87ec86e927ef0be140bfdb0d2"}, +    {file = "cffi-1.14.6.tar.gz", hash = "sha256:c9a875ce9d7fe32887784274dd533c57909b7b1dcadcc128a2ac21331a9765dd"},  ]  cfgv = [      {file = "cfgv-3.3.0-py2.py3-none-any.whl", hash = "sha256:b449c9c6118fe8cca7fa5e00b9ec60ba08145d281d52164230a69211c5d597a1"}, @@ -1156,6 +1193,10 @@ chardet = [      {file = "chardet-4.0.0-py2.py3-none-any.whl", hash = "sha256:f864054d66fd9118f2e67044ac8981a54775ec5b67aed0441892edb553d21da5"},      {file = "chardet-4.0.0.tar.gz", hash = "sha256:0d6f53a15db4120f2b08c94f11e7d93d2c911ee118b6b30a04ec3ee8310179fa"},  ] +charset-normalizer = [ +    {file = "charset-normalizer-2.0.4.tar.gz", hash = "sha256:f23667ebe1084be45f6ae0538e4a5a865206544097e4e8bbcacf42cd02a348f3"}, +    {file = "charset_normalizer-2.0.4-py3-none-any.whl", hash = "sha256:0c8911edd15d19223366a194a513099a302055a962bca2cec0f54b8b63175d8b"}, +]  colorama = [      {file = "colorama-0.4.4-py2.py3-none-any.whl", hash = "sha256:9f47eda37229f68eee03b24b9748937c7dc3868f906e8ba69fbcbdd3bc5dc3e2"},      {file = "colorama-0.4.4.tar.gz", hash = "sha256:5941b2b48a20143d2267e95b1c2a7603ce057ee39fd88e7329b0c292aa16869b"}, @@ -1339,12 +1380,12 @@ humanfriendly = [      {file = "humanfriendly-9.2.tar.gz", hash = "sha256:f7dba53ac7935fd0b4a2fc9a29e316ddd9ea135fb3052d3d0279d10c18ff9c48"},  ]  identify = [ -    {file = "identify-2.2.10-py2.py3-none-any.whl", hash = "sha256:18d0c531ee3dbc112fa6181f34faa179de3f57ea57ae2899754f16a7e0ff6421"}, -    {file = "identify-2.2.10.tar.gz", hash = "sha256:5b41f71471bc738e7b586308c3fca172f78940195cb3bf6734c1e66fdac49306"}, +    {file = "identify-2.2.11-py2.py3-none-any.whl", hash = "sha256:7abaecbb414e385752e8ce02d8c494f4fbc780c975074b46172598a28f1ab839"}, +    {file = "identify-2.2.11.tar.gz", hash = "sha256:a0e700637abcbd1caae58e0463861250095dfe330a8371733a471af706a4a29a"},  ]  idna = [ -    {file = "idna-2.10-py2.py3-none-any.whl", hash = "sha256:b97d804b1e9b523befed77c48dacec60e6dcb0b5391d57af6a65a312a90648c0"}, -    {file = "idna-2.10.tar.gz", hash = "sha256:b307872f855b18632ce0c21c5e45be78c0ea7ae4c15c828c20788b26921eb3f6"}, +    {file = "idna-3.2-py3-none-any.whl", hash = "sha256:14475042e284991034cb48e06f6851428fb14c4dc953acd9be9a5e95c7b6dd7a"}, +    {file = "idna-3.2.tar.gz", hash = "sha256:467fbad99067910785144ce333826c71fb0e63a425657295239737f7ecd125f3"},  ]  iniconfig = [      {file = "iniconfig-1.1.1-py2.py3-none-any.whl", hash = "sha256:011e24c64b7f47f6ebd835bb12a743f2fbe9a26d4cecaa7f53bc4f35ee9da8b3"}, @@ -1472,6 +1513,10 @@ pep8-naming = [      {file = "pep8-naming-0.12.0.tar.gz", hash = "sha256:1f9a3ecb2f3fd83240fd40afdd70acc89695c49c333413e49788f93b61827e12"},      {file = "pep8_naming-0.12.0-py2.py3-none-any.whl", hash = "sha256:2321ac2b7bf55383dd19a6a9c8ae2ebf05679699927a3af33e60dd7d337099d3"},  ] +platformdirs = [ +    {file = "platformdirs-2.2.0-py3-none-any.whl", hash = "sha256:4666d822218db6a262bdfdc9c39d21f23b4cfdb08af331a81e92751daf6c866c"}, +    {file = "platformdirs-2.2.0.tar.gz", hash = "sha256:632daad3ab546bd8e6af0537d09805cec458dce201bccfe23012df73332e181e"}, +]  pluggy = [      {file = "pluggy-0.13.1-py2.py3-none-any.whl", hash = "sha256:966c145cd83c96502c3c3868f50408687b38434af77734af1e9ca461a4081d2d"},      {file = "pluggy-0.13.1.tar.gz", hash = "sha256:15b2acde666561e1298d71b523007ed7364de07029219b604cf808bfa1c765b0"}, @@ -1591,8 +1636,8 @@ pytest-xdist = [      {file = "pytest_xdist-2.3.0-py3-none-any.whl", hash = "sha256:ed3d7da961070fce2a01818b51f6888327fb88df4379edeb6b9d990e789d9c8d"},  ]  python-dateutil = [ -    {file = "python-dateutil-2.8.1.tar.gz", hash = "sha256:73ebfe9dbf22e832286dafa60473e4cd239f8592f699aa5adaf10050e6e1823c"}, -    {file = "python_dateutil-2.8.1-py2.py3-none-any.whl", hash = "sha256:75bb3f31ea686f1197762692a9ee6a7550b59fc6ca3a1f4b5d7e32fb98e2da2a"}, +    {file = "python-dateutil-2.8.2.tar.gz", hash = "sha256:0123cacc1627ae19ddf3c27a5de5bd67ee4586fbdd6440d9748f8abb483d3e86"}, +    {file = "python_dateutil-2.8.2-py2.py3-none-any.whl", hash = "sha256:961d03dc3453ebbc59dbdea9e4e11c5651520a876d0f4db161e8674aae935da9"},  ]  python-dotenv = [      {file = "python-dotenv-0.17.1.tar.gz", hash = "sha256:b1ae5e9643d5ed987fc57cc2583021e38db531946518130777734f9589b3141f"}, @@ -1609,18 +1654,26 @@ pyyaml = [      {file = "PyYAML-5.4.1-cp27-cp27mu-manylinux1_x86_64.whl", hash = "sha256:bb4191dfc9306777bc594117aee052446b3fa88737cd13b7188d0e7aa8162185"},      {file = "PyYAML-5.4.1-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:6c78645d400265a062508ae399b60b8c167bf003db364ecb26dcab2bda048253"},      {file = "PyYAML-5.4.1-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:4e0583d24c881e14342eaf4ec5fbc97f934b999a6828693a99157fde912540cc"}, +    {file = "PyYAML-5.4.1-cp36-cp36m-manylinux2014_aarch64.whl", hash = "sha256:72a01f726a9c7851ca9bfad6fd09ca4e090a023c00945ea05ba1638c09dc3347"}, +    {file = "PyYAML-5.4.1-cp36-cp36m-manylinux2014_s390x.whl", hash = "sha256:895f61ef02e8fed38159bb70f7e100e00f471eae2bc838cd0f4ebb21e28f8541"},      {file = "PyYAML-5.4.1-cp36-cp36m-win32.whl", hash = "sha256:3bd0e463264cf257d1ffd2e40223b197271046d09dadf73a0fe82b9c1fc385a5"},      {file = "PyYAML-5.4.1-cp36-cp36m-win_amd64.whl", hash = "sha256:e4fac90784481d221a8e4b1162afa7c47ed953be40d31ab4629ae917510051df"},      {file = "PyYAML-5.4.1-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:5accb17103e43963b80e6f837831f38d314a0495500067cb25afab2e8d7a4018"},      {file = "PyYAML-5.4.1-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:e1d4970ea66be07ae37a3c2e48b5ec63f7ba6804bdddfdbd3cfd954d25a82e63"}, +    {file = "PyYAML-5.4.1-cp37-cp37m-manylinux2014_aarch64.whl", hash = "sha256:cb333c16912324fd5f769fff6bc5de372e9e7a202247b48870bc251ed40239aa"}, +    {file = "PyYAML-5.4.1-cp37-cp37m-manylinux2014_s390x.whl", hash = "sha256:fe69978f3f768926cfa37b867e3843918e012cf83f680806599ddce33c2c68b0"},      {file = "PyYAML-5.4.1-cp37-cp37m-win32.whl", hash = "sha256:dd5de0646207f053eb0d6c74ae45ba98c3395a571a2891858e87df7c9b9bd51b"},      {file = "PyYAML-5.4.1-cp37-cp37m-win_amd64.whl", hash = "sha256:08682f6b72c722394747bddaf0aa62277e02557c0fd1c42cb853016a38f8dedf"},      {file = "PyYAML-5.4.1-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:d2d9808ea7b4af864f35ea216be506ecec180628aced0704e34aca0b040ffe46"},      {file = "PyYAML-5.4.1-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:8c1be557ee92a20f184922c7b6424e8ab6691788e6d86137c5d93c1a6ec1b8fb"}, +    {file = "PyYAML-5.4.1-cp38-cp38-manylinux2014_aarch64.whl", hash = "sha256:fd7f6999a8070df521b6384004ef42833b9bd62cfee11a09bda1079b4b704247"}, +    {file = "PyYAML-5.4.1-cp38-cp38-manylinux2014_s390x.whl", hash = "sha256:bfb51918d4ff3d77c1c856a9699f8492c612cde32fd3bcd344af9be34999bfdc"},      {file = "PyYAML-5.4.1-cp38-cp38-win32.whl", hash = "sha256:fa5ae20527d8e831e8230cbffd9f8fe952815b2b7dae6ffec25318803a7528fc"},      {file = "PyYAML-5.4.1-cp38-cp38-win_amd64.whl", hash = "sha256:0f5f5786c0e09baddcd8b4b45f20a7b5d61a7e7e99846e3c799b05c7c53fa696"},      {file = "PyYAML-5.4.1-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:294db365efa064d00b8d1ef65d8ea2c3426ac366c0c4368d930bf1c5fb497f77"},      {file = "PyYAML-5.4.1-cp39-cp39-manylinux1_x86_64.whl", hash = "sha256:74c1485f7707cf707a7aef42ef6322b8f97921bd89be2ab6317fd782c2d53183"}, +    {file = "PyYAML-5.4.1-cp39-cp39-manylinux2014_aarch64.whl", hash = "sha256:d483ad4e639292c90170eb6f7783ad19490e7a8defb3e46f97dfe4bacae89122"}, +    {file = "PyYAML-5.4.1-cp39-cp39-manylinux2014_s390x.whl", hash = "sha256:fdc842473cd33f45ff6bce46aea678a54e3d21f1b61a7750ce3c498eedfe25d6"},      {file = "PyYAML-5.4.1-cp39-cp39-win32.whl", hash = "sha256:49d4cdd9065b9b6e206d0595fee27a96b5dd22618e7520c33204a4a3239d5b10"},      {file = "PyYAML-5.4.1-cp39-cp39-win_amd64.whl", hash = "sha256:c20cfa2d49991c8b4147af39859b167664f2ad4561704ee74c1de03318e898db"},      {file = "PyYAML-5.4.1.tar.gz", hash = "sha256:607774cbba28732bfa802b54baa7484215f530991055bb562efbed5b2f20a45e"}, @@ -1736,12 +1789,12 @@ regex = [      {file = "regex-2021.4.4.tar.gz", hash = "sha256:52ba3d3f9b942c49d7e4bc105bb28551c44065f139a65062ab7912bef10c9afb"},  ]  requests = [ -    {file = "requests-2.25.1-py2.py3-none-any.whl", hash = "sha256:c210084e36a42ae6b9219e00e48287def368a26d03a048ddad7bfee44f75871e"}, -    {file = "requests-2.25.1.tar.gz", hash = "sha256:27973dd4a904a4f13b263a19c866c13b92a39ed1c964655f025f3f8d3d75b804"}, +    {file = "requests-2.26.0-py2.py3-none-any.whl", hash = "sha256:6c1246513ecd5ecd4528a0906f910e8f0f9c6b8ec72030dc9fd154dc1a6efd24"}, +    {file = "requests-2.26.0.tar.gz", hash = "sha256:b8aa58f8cf793ffd8782d3d8cb19e66ef36f7aba4353eec859e74678b01b07a7"},  ]  sentry-sdk = [ -    {file = "sentry-sdk-0.20.3.tar.gz", hash = "sha256:4ae8d1ced6c67f1c8ea51d82a16721c166c489b76876c9f2c202b8a50334b237"}, -    {file = "sentry_sdk-0.20.3-py2.py3-none-any.whl", hash = "sha256:e75c8c58932bda8cd293ea8e4b242527129e1caaec91433d21b8b2f20fee030b"}, +    {file = "sentry-sdk-1.3.1.tar.gz", hash = "sha256:ebe99144fa9618d4b0e7617e7929b75acd905d258c3c779edcd34c0adfffe26c"}, +    {file = "sentry_sdk-1.3.1-py2.py3-none-any.whl", hash = "sha256:f33d34c886d0ba24c75ea8885a8b3a172358853c7cbde05979fc99c29ef7bc52"},  ]  sgmllib3k = [      {file = "sgmllib3k-1.0.0.tar.gz", hash = "sha256:7868fb1c8bfa764c1ac563d3cf369c381d1325d36124933a726f29fcdaa812e9"}, @@ -1784,8 +1837,8 @@ urllib3 = [      {file = "urllib3-1.26.6.tar.gz", hash = "sha256:f57b4c16c62fa2760b7e3d97c35b255512fb6b59a259730f36ba32ce9f8e342f"},  ]  virtualenv = [ -    {file = "virtualenv-20.4.7-py2.py3-none-any.whl", hash = "sha256:2b0126166ea7c9c3661f5b8e06773d28f83322de7a3ff7d06f0aed18c9de6a76"}, -    {file = "virtualenv-20.4.7.tar.gz", hash = "sha256:14fdf849f80dbb29a4eb6caa9875d476ee2a5cf76a5f5415fa2f1606010ab467"}, +    {file = "virtualenv-20.7.0-py2.py3-none-any.whl", hash = "sha256:fdfdaaf0979ac03ae7f76d5224a05b58165f3c804f8aa633f3dd6f22fbd435d5"}, +    {file = "virtualenv-20.7.0.tar.gz", hash = "sha256:97066a978431ec096d163e72771df5357c5c898ffdd587048f45e0aecc228094"},  ]  yarl = [      {file = "yarl-1.6.3-cp36-cp36m-macosx_10_14_x86_64.whl", hash = "sha256:0355a701b3998dcd832d0dc47cc5dedf3874f966ac7f870e0f3a6788d802d434"}, diff --git a/pyproject.toml b/pyproject.toml index 8eac504c5..2ae79f9e4 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -28,7 +28,7 @@ python-dateutil = "~=2.8"  python-frontmatter = "~=1.0.0"  pyyaml = "~=5.1"  regex = "==2021.4.4" -sentry-sdk = "~=0.19" +sentry-sdk = "~=1.3"  statsd = "~=3.3"  [tool.poetry.dev-dependencies] diff --git a/tests/bot/exts/events/__init__.py b/tests/bot/exts/events/__init__.py new file mode 100644 index 000000000..e69de29bb --- /dev/null +++ b/tests/bot/exts/events/__init__.py diff --git a/tests/bot/exts/utils/test_jams.py b/tests/bot/exts/events/test_code_jams.py index 368a15476..b9ee1e363 100644 --- a/tests/bot/exts/utils/test_jams.py +++ b/tests/bot/exts/events/test_code_jams.py @@ -1,14 +1,15 @@  import unittest -from unittest.mock import AsyncMock, MagicMock, create_autospec +from unittest.mock import AsyncMock, MagicMock, create_autospec, patch  from discord import CategoryChannel  from discord.ext.commands import BadArgument  from bot.constants import Roles -from bot.exts.utils import jams +from bot.exts.events import code_jams +from bot.exts.events.code_jams import _channels, _cog  from tests.helpers import (      MockAttachment, MockBot, MockCategoryChannel, MockContext, -    MockGuild, MockMember, MockRole, MockTextChannel +    MockGuild, MockMember, MockRole, MockTextChannel, autospec  )  TEST_CSV = b"""\ @@ -40,7 +41,7 @@ class JamCodejamCreateTests(unittest.IsolatedAsyncioTestCase):          self.command_user = MockMember([self.admin_role])          self.guild = MockGuild([self.admin_role])          self.ctx = MockContext(bot=self.bot, author=self.command_user, guild=self.guild) -        self.cog = jams.CodeJams(self.bot) +        self.cog = _cog.CodeJams(self.bot)      async def test_message_without_attachments(self):          """If no link or attachments are provided, commands.BadArgument should be raised.""" @@ -49,7 +50,9 @@ class JamCodejamCreateTests(unittest.IsolatedAsyncioTestCase):          with self.assertRaises(BadArgument):              await self.cog.create(self.cog, self.ctx, None) -    async def test_result_sending(self): +    @patch.object(_channels, "create_team_channel") +    @patch.object(_channels, "create_team_leader_channel") +    async def test_result_sending(self, create_leader_channel, create_team_channel):          """Should call `ctx.send` when everything goes right."""          self.ctx.message.attachments = [MockAttachment()]          self.ctx.message.attachments[0].read = AsyncMock() @@ -61,14 +64,12 @@ class JamCodejamCreateTests(unittest.IsolatedAsyncioTestCase):          self.ctx.guild.create_role = AsyncMock()          self.ctx.guild.create_role.return_value = team_leaders -        self.cog.create_team_channel = AsyncMock() -        self.cog.create_team_leader_channel = AsyncMock()          self.cog.add_roles = AsyncMock()          await self.cog.create(self.cog, self.ctx, None) -        self.cog.create_team_channel.assert_awaited() -        self.cog.create_team_leader_channel.assert_awaited_once_with( +        create_team_channel.assert_awaited() +        create_leader_channel.assert_awaited_once_with(              self.ctx.guild, team_leaders          )          self.ctx.send.assert_awaited_once() @@ -81,25 +82,24 @@ class JamCodejamCreateTests(unittest.IsolatedAsyncioTestCase):          self.ctx.send.assert_awaited_once() -    async def test_category_doesnt_exist(self): +    @patch.object(_channels, "_send_status_update") +    async def test_category_doesnt_exist(self, update):          """Should create a new code jam category."""          subtests = (              [], -            [get_mock_category(jams.MAX_CHANNELS, jams.CATEGORY_NAME)], -            [get_mock_category(jams.MAX_CHANNELS - 2, "other")], +            [get_mock_category(_channels.MAX_CHANNELS, _channels.CATEGORY_NAME)], +            [get_mock_category(_channels.MAX_CHANNELS - 2, "other")],          ) -        self.cog.send_status_update = AsyncMock() -          for categories in subtests: -            self.cog.send_status_update.reset_mock() +            update.reset_mock()              self.guild.reset_mock()              self.guild.categories = categories              with self.subTest(categories=categories): -                actual_category = await self.cog.get_category(self.guild) +                actual_category = await _channels._get_category(self.guild) -                self.cog.send_status_update.assert_called_once() +                update.assert_called_once()                  self.guild.create_category_channel.assert_awaited_once()                  category_overwrites = self.guild.create_category_channel.call_args[1]["overwrites"] @@ -109,45 +109,41 @@ class JamCodejamCreateTests(unittest.IsolatedAsyncioTestCase):      async def test_category_channel_exist(self):          """Should not try to create category channel.""" -        expected_category = get_mock_category(jams.MAX_CHANNELS - 2, jams.CATEGORY_NAME) +        expected_category = get_mock_category(_channels.MAX_CHANNELS - 2, _channels.CATEGORY_NAME)          self.guild.categories = [ -            get_mock_category(jams.MAX_CHANNELS - 2, "other"), +            get_mock_category(_channels.MAX_CHANNELS - 2, "other"),              expected_category, -            get_mock_category(0, jams.CATEGORY_NAME), +            get_mock_category(0, _channels.CATEGORY_NAME),          ] -        actual_category = await self.cog.get_category(self.guild) +        actual_category = await _channels._get_category(self.guild)          self.assertEqual(expected_category, actual_category)      async def test_channel_overwrites(self):          """Should have correct permission overwrites for users and roles."""          leader = (MockMember(), True)          members = [leader] + [(MockMember(), False) for _ in range(4)] -        overwrites = self.cog.get_overwrites(members, self.guild) +        overwrites = _channels._get_overwrites(members, self.guild)          for member, _ in members:              self.assertTrue(overwrites[member].read_messages) -    async def test_team_channels_creation(self): +    @patch.object(_channels, "_get_overwrites") +    @patch.object(_channels, "_get_category") +    @autospec(_channels, "_add_team_leader_roles", pass_mocks=False) +    async def test_team_channels_creation(self, get_category, get_overwrites):          """Should create a text channel for a team."""          team_leaders = MockRole()          members = [(MockMember(), True)] + [(MockMember(), False) for _ in range(5)]          category = MockCategoryChannel()          category.create_text_channel = AsyncMock() -        self.cog.get_overwrites = MagicMock() -        self.cog.get_category = AsyncMock() -        self.cog.get_category.return_value = category -        self.cog.add_team_leader_roles = AsyncMock() - -        await self.cog.create_team_channel(self.guild, "my-team", members, team_leaders) -        self.cog.add_team_leader_roles.assert_awaited_once_with(members, team_leaders) -        self.cog.get_overwrites.assert_called_once_with(members, self.guild) -        self.cog.get_category.assert_awaited_once_with(self.guild) +        get_category.return_value = category +        await _channels.create_team_channel(self.guild, "my-team", members, team_leaders)          category.create_text_channel.assert_awaited_once_with(              "my-team", -            overwrites=self.cog.get_overwrites.return_value +            overwrites=get_overwrites.return_value          )      async def test_jam_roles_adding(self): @@ -156,7 +152,7 @@ class JamCodejamCreateTests(unittest.IsolatedAsyncioTestCase):          leader = MockMember()          members = [(leader, True)] + [(MockMember(), False) for _ in range(4)] -        await self.cog.add_team_leader_roles(members, leader_role) +        await _channels._add_team_leader_roles(members, leader_role)          leader.add_roles.assert_awaited_once_with(leader_role)          for member, is_leader in members: @@ -170,5 +166,5 @@ class CodeJamSetup(unittest.TestCase):      def test_setup(self):          """Should call `bot.add_cog`."""          bot = MockBot() -        jams.setup(bot) +        code_jams.setup(bot)          bot.add_cog.assert_called_once() diff --git a/tests/bot/exts/info/test_information.py b/tests/bot/exts/info/test_information.py index 770660fe3..0aa41d889 100644 --- a/tests/bot/exts/info/test_information.py +++ b/tests/bot/exts/info/test_information.py @@ -262,7 +262,6 @@ class UserInfractionHelperMethodTests(unittest.IsolatedAsyncioTestCase):          await self._method_subtests(self.cog.user_nomination_counts, test_values, header) [email protected]("bot.exts.info.information.time_since", new=unittest.mock.MagicMock(return_value="1 year ago"))  @unittest.mock.patch("bot.exts.info.information.constants.MODERATION_CHANNELS", new=[50])  class UserEmbedTests(unittest.IsolatedAsyncioTestCase):      """Tests for the creation of the `!user` embed.""" @@ -347,7 +346,7 @@ class UserEmbedTests(unittest.IsolatedAsyncioTestCase):          self.assertEqual(              textwrap.dedent(f""" -                Created: {"1 year ago"} +                Created: {"<t:1:R>"}                  Profile: {user.mention}                  ID: {user.id}              """).strip(), @@ -356,7 +355,7 @@ class UserEmbedTests(unittest.IsolatedAsyncioTestCase):          self.assertEqual(              textwrap.dedent(f""" -                Joined: {"1 year ago"} +                Joined: {"<t:1:R>"}                  Verified: {"True"}                  Roles: &Moderators              """).strip(), @@ -379,7 +378,7 @@ class UserEmbedTests(unittest.IsolatedAsyncioTestCase):          self.assertEqual(              textwrap.dedent(f""" -                Created: {"1 year ago"} +                Created: {"<t:1:R>"}                  Profile: {user.mention}                  ID: {user.id}              """).strip(), @@ -388,7 +387,7 @@ class UserEmbedTests(unittest.IsolatedAsyncioTestCase):          self.assertEqual(              textwrap.dedent(f""" -                Joined: {"1 year ago"} +                Joined: {"<t:1:R>"}                  Roles: &Moderators              """).strip(),              embed.fields[1].value diff --git a/tests/bot/exts/moderation/infraction/test_utils.py b/tests/bot/exts/moderation/infraction/test_utils.py index f3af7bea9..eb256f1fd 100644 --- a/tests/bot/exts/moderation/infraction/test_utils.py +++ b/tests/bot/exts/moderation/infraction/test_utils.py @@ -137,7 +137,7 @@ class ModerationUtilsTests(unittest.IsolatedAsyncioTestCase):                      title=utils.INFRACTION_TITLE,                      description=utils.INFRACTION_DESCRIPTION_TEMPLATE.format(                          type="Ban", -                        expires="2020-02-26 09:20 (23 hours and 59 minutes) UTC", +                        expires="2020-02-26 09:20 (23 hours and 59 minutes)",                          reason="No reason provided."                      ),                      colour=Colours.soft_red, @@ -193,7 +193,7 @@ class ModerationUtilsTests(unittest.IsolatedAsyncioTestCase):                      title=utils.INFRACTION_TITLE,                      description=utils.INFRACTION_DESCRIPTION_TEMPLATE.format(                          type="Mute", -                        expires="2020-02-26 09:20 (23 hours and 59 minutes) UTC", +                        expires="2020-02-26 09:20 (23 hours and 59 minutes)",                          reason="Test"                      ),                      colour=Colours.soft_red, @@ -213,7 +213,7 @@ class ModerationUtilsTests(unittest.IsolatedAsyncioTestCase):                          type="Mute",                          expires="N/A",                          reason="foo bar" * 4000 -                    )[:2045] + "...", +                    )[:4093] + "...",                      colour=Colours.soft_red,                      url=utils.RULES_URL                  ).set_author( diff --git a/tests/bot/exts/moderation/test_modlog.py b/tests/bot/exts/moderation/test_modlog.py index f8f142484..79e04837d 100644 --- a/tests/bot/exts/moderation/test_modlog.py +++ b/tests/bot/exts/moderation/test_modlog.py @@ -25,5 +25,5 @@ class ModLogTests(unittest.IsolatedAsyncioTestCase):          )          embed = self.channel.send.call_args[1]["embed"]          self.assertEqual( -            embed.description, ("foo bar" * 3000)[:2045] + "..." +            embed.description, ("foo bar" * 3000)[:4093] + "..."          ) diff --git a/tests/bot/utils/test_time.py b/tests/bot/utils/test_time.py index 115ddfb0d..8edffd1c9 100644 --- a/tests/bot/utils/test_time.py +++ b/tests/bot/utils/test_time.py @@ -52,7 +52,7 @@ class TimeTests(unittest.TestCase):      def test_format_infraction(self):          """Testing format_infraction.""" -        self.assertEqual(time.format_infraction('2019-12-12T00:01:00Z'), '2019-12-12 00:01') +        self.assertEqual(time.format_infraction('2019-12-12T00:01:00Z'), '<t:1576108860:f>')      def test_format_infraction_with_duration_none_expiry(self):          """format_infraction_with_duration should work for None expiry.""" @@ -72,10 +72,10 @@ class TimeTests(unittest.TestCase):      def test_format_infraction_with_duration_custom_units(self):          """format_infraction_with_duration should work for custom max_units."""          test_cases = ( -            ('2019-12-12T00:01:00Z', datetime(2019, 12, 11, 12, 5, 5), 6, -             '2019-12-12 00:01 (11 hours, 55 minutes and 55 seconds)'), -            ('2019-11-23T20:09:00Z', datetime(2019, 4, 25, 20, 15), 20, -             '2019-11-23 20:09 (6 months, 28 days, 23 hours and 54 minutes)') +            ('3000-12-12T00:01:00Z', datetime(3000, 12, 11, 12, 5, 5), 6, +             '<t:32533488060:f> (11 hours, 55 minutes and 55 seconds)'), +            ('3000-11-23T20:09:00Z', datetime(3000, 4, 25, 20, 15), 20, +             '<t:32531918940:f> (6 months, 28 days, 23 hours and 54 minutes)')          )          for expiry, date_from, max_units, expected in test_cases: @@ -85,16 +85,16 @@ class TimeTests(unittest.TestCase):      def test_format_infraction_with_duration_normal_usage(self):          """format_infraction_with_duration should work for normal usage, across various durations."""          test_cases = ( -            ('2019-12-12T00:01:00Z', datetime(2019, 12, 11, 12, 0, 5), 2, '2019-12-12 00:01 (12 hours and 55 seconds)'), -            ('2019-12-12T00:01:00Z', datetime(2019, 12, 11, 12, 0, 5), 1, '2019-12-12 00:01 (12 hours)'), -            ('2019-12-12T00:00:00Z', datetime(2019, 12, 11, 23, 59), 2, '2019-12-12 00:00 (1 minute)'), -            ('2019-11-23T20:09:00Z', datetime(2019, 11, 15, 20, 15), 2, '2019-11-23 20:09 (7 days and 23 hours)'), -            ('2019-11-23T20:09:00Z', datetime(2019, 4, 25, 20, 15), 2, '2019-11-23 20:09 (6 months and 28 days)'), -            ('2019-11-23T20:58:00Z', datetime(2019, 11, 23, 20, 53), 2, '2019-11-23 20:58 (5 minutes)'), -            ('2019-11-24T00:00:00Z', datetime(2019, 11, 23, 23, 59, 0), 2, '2019-11-24 00:00 (1 minute)'), -            ('2019-11-23T23:59:00Z', datetime(2017, 7, 21, 23, 0), 2, '2019-11-23 23:59 (2 years and 4 months)'), +            ('2019-12-12T00:01:00Z', datetime(2019, 12, 11, 12, 0, 5), 2, '<t:1576108860:f> (12 hours and 55 seconds)'), +            ('2019-12-12T00:01:00Z', datetime(2019, 12, 11, 12, 0, 5), 1, '<t:1576108860:f> (12 hours)'), +            ('2019-12-12T00:00:00Z', datetime(2019, 12, 11, 23, 59), 2, '<t:1576108800:f> (1 minute)'), +            ('2019-11-23T20:09:00Z', datetime(2019, 11, 15, 20, 15), 2, '<t:1574539740:f> (7 days and 23 hours)'), +            ('2019-11-23T20:09:00Z', datetime(2019, 4, 25, 20, 15), 2, '<t:1574539740:f> (6 months and 28 days)'), +            ('2019-11-23T20:58:00Z', datetime(2019, 11, 23, 20, 53), 2, '<t:1574542680:f> (5 minutes)'), +            ('2019-11-24T00:00:00Z', datetime(2019, 11, 23, 23, 59, 0), 2, '<t:1574553600:f> (1 minute)'), +            ('2019-11-23T23:59:00Z', datetime(2017, 7, 21, 23, 0), 2, '<t:1574553540:f> (2 years and 4 months)'),              ('2019-11-23T23:59:00Z', datetime(2019, 11, 23, 23, 49, 5), 2, -             '2019-11-23 23:59 (9 minutes and 55 seconds)'), +             '<t:1574553540:f> (9 minutes and 55 seconds)'),              (None, datetime(2019, 11, 23, 23, 49, 5), 2, None),          ) @@ -104,45 +104,30 @@ class TimeTests(unittest.TestCase):      def test_until_expiration_with_duration_none_expiry(self):          """until_expiration should work for None expiry.""" -        test_cases = ( -            (None, None, None, None), - -            # To make sure that now and max_units are not touched -            (None, 'Why hello there!', None, None), -            (None, None, float('inf'), None), -            (None, 'Why hello there!', float('inf'), None), -        ) - -        for expiry, now, max_units, expected in test_cases: -            with self.subTest(expiry=expiry, now=now, max_units=max_units, expected=expected): -                self.assertEqual(time.until_expiration(expiry, now, max_units), expected) +        self.assertEqual(time.until_expiration(None), None)      def test_until_expiration_with_duration_custom_units(self):          """until_expiration should work for custom max_units."""          test_cases = ( -            ('2019-12-12T00:01:00Z', datetime(2019, 12, 11, 12, 5, 5), 6, '11 hours, 55 minutes and 55 seconds'), -            ('2019-11-23T20:09:00Z', datetime(2019, 4, 25, 20, 15), 20, '6 months, 28 days, 23 hours and 54 minutes') +            ('3000-12-12T00:01:00Z', '<t:32533488060:R>'), +            ('3000-11-23T20:09:00Z', '<t:32531918940:R>')          ) -        for expiry, now, max_units, expected in test_cases: -            with self.subTest(expiry=expiry, now=now, max_units=max_units, expected=expected): -                self.assertEqual(time.until_expiration(expiry, now, max_units), expected) +        for expiry, expected in test_cases: +            with self.subTest(expiry=expiry, expected=expected): +                self.assertEqual(time.until_expiration(expiry,), expected)      def test_until_expiration_normal_usage(self):          """until_expiration should work for normal usage, across various durations."""          test_cases = ( -            ('2019-12-12T00:01:00Z', datetime(2019, 12, 11, 12, 0, 5), 2, '12 hours and 55 seconds'), -            ('2019-12-12T00:01:00Z', datetime(2019, 12, 11, 12, 0, 5), 1, '12 hours'), -            ('2019-12-12T00:00:00Z', datetime(2019, 12, 11, 23, 59), 2, '1 minute'), -            ('2019-11-23T20:09:00Z', datetime(2019, 11, 15, 20, 15), 2, '7 days and 23 hours'), -            ('2019-11-23T20:09:00Z', datetime(2019, 4, 25, 20, 15), 2, '6 months and 28 days'), -            ('2019-11-23T20:58:00Z', datetime(2019, 11, 23, 20, 53), 2, '5 minutes'), -            ('2019-11-24T00:00:00Z', datetime(2019, 11, 23, 23, 59, 0), 2, '1 minute'), -            ('2019-11-23T23:59:00Z', datetime(2017, 7, 21, 23, 0), 2, '2 years and 4 months'), -            ('2019-11-23T23:59:00Z', datetime(2019, 11, 23, 23, 49, 5), 2, '9 minutes and 55 seconds'), -            (None, datetime(2019, 11, 23, 23, 49, 5), 2, None), +            ('3000-12-12T00:01:00Z', '<t:32533488060:R>'), +            ('3000-12-12T00:01:00Z', '<t:32533488060:R>'), +            ('3000-12-12T00:00:00Z', '<t:32533488000:R>'), +            ('3000-11-23T20:09:00Z', '<t:32531918940:R>'), +            ('3000-11-23T20:09:00Z', '<t:32531918940:R>'), +            (None, None),          ) -        for expiry, now, max_units, expected in test_cases: -            with self.subTest(expiry=expiry, now=now, max_units=max_units, expected=expected): -                self.assertEqual(time.until_expiration(expiry, now, max_units), expected) +        for expiry, expected in test_cases: +            with self.subTest(expiry=expiry, expected=expected): +                self.assertEqual(time.until_expiration(expiry), expected) | 
