diff options
| author | 2019-10-04 13:57:22 -0700 | |
|---|---|---|
| committer | 2019-10-04 13:57:22 -0700 | |
| commit | b8e38b2090fa0fa02228da4fc36c9f42d10ebd17 (patch) | |
| tree | fe3a81d2f6045ae1ab73ac05a4fd8c5290bb4f56 | |
| parent | Check if tzinfo is None in ISODateTime test (diff) | |
| parent | Merge pull request #481 from bendiller/fetch_posts_retries (diff) | |
Merge branch 'master' into ISODate-converter
| -rw-r--r-- | CONTRIBUTING.md | 6 | ||||
| -rw-r--r-- | bot/cogs/alias.py | 5 | ||||
| -rw-r--r-- | bot/cogs/doc.py | 8 | ||||
| -rw-r--r-- | bot/cogs/moderation.py | 47 | ||||
| -rw-r--r-- | bot/cogs/reddit.py | 39 | ||||
| -rw-r--r-- | bot/cogs/site.py | 26 | ||||
| -rw-r--r-- | bot/cogs/tags.py | 33 | ||||
| -rw-r--r-- | bot/cogs/watchchannels/watchchannel.py | 2 | ||||
| -rw-r--r-- | bot/converters.py | 2 | 
9 files changed, 130 insertions, 38 deletions
| diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md index a0a1200ec..39f76c7b4 100644 --- a/CONTRIBUTING.md +++ b/CONTRIBUTING.md @@ -36,9 +36,9 @@ All projects evolve over time, and this contribution guide is no different. This  ##  Supplemental Information  ### Developer Environment -A working environment for the [PyDis site](https://github.com/python-discord/site) is required to develop the bot. Instructions for setting up environments for both the site and the bot can be found on the PyDis Wiki: -  * [Site](https://wiki.pythondiscord.com/wiki/contributing/project/site) -  * [Bot](https://wiki.pythondiscord.com/wiki/contributing/project/bot) +Instructions for setting the bot developer environment can be found on the [PyDis wiki](https://pythondiscord.com/pages/contributing/bot/) + +To provide a standalone development environment for this project, docker compose is utilized to pull the current version of the [site backend](https://github.com/python-discord/site). While appropriate for bot-only contributions, any contributions that necessitate backend changes will require the site repository to be appropriately configured as well. Instructions for setting up the site environment can be found on the [PyDis site](https://pythondiscord.com/pages/contributing/site/).  When pulling down changes from GitHub, remember to sync your environment using `pipenv sync --dev` to ensure you're using the most up-to-date versions the project's dependencies. diff --git a/bot/cogs/alias.py b/bot/cogs/alias.py index 80ff37983..0f49a400c 100644 --- a/bot/cogs/alias.py +++ b/bot/cogs/alias.py @@ -53,6 +53,11 @@ class Alias (Cog):          """Alias for invoking <prefix>site resources."""          await self.invoke(ctx, "site resources") +    @command(name="tools", hidden=True) +    async def site_tools_alias(self, ctx: Context) -> None: +        """Alias for invoking <prefix>site tools.""" +        await self.invoke(ctx, "site tools") +      @command(name="watch", hidden=True)      async def bigbrother_watch_alias(self, ctx: Context, user: Union[Member, User, proxy_user], *, reason: str) -> None:          """Alias for invoking <prefix>bigbrother watch [user] [reason].""" diff --git a/bot/cogs/doc.py b/bot/cogs/doc.py index e5c51748f..c9e6b3b91 100644 --- a/bot/cogs/doc.py +++ b/bot/cogs/doc.py @@ -261,7 +261,7 @@ class Doc(commands.Cog):      @commands.group(name='docs', aliases=('doc', 'd'), invoke_without_command=True)      async def docs_group(self, ctx: commands.Context, symbol: commands.clean_content = None) -> None:          """Lookup documentation for Python symbols.""" -        await ctx.invoke(self.get_command) +        await ctx.invoke(self.get_command, symbol)      @docs_group.command(name='get', aliases=('g',))      async def get_command(self, ctx: commands.Context, symbol: commands.clean_content = None) -> None: @@ -319,9 +319,9 @@ class Doc(commands.Cog):          Example:              !docs set \ -                    discord \ -                    https://discordpy.readthedocs.io/en/rewrite/ \ -                    https://discordpy.readthedocs.io/en/rewrite/objects.inv +                    python \ +                    https://docs.python.org/3/ \ +                    https://docs.python.org/3/objects.inv          """          body = {              'package': package_name, diff --git a/bot/cogs/moderation.py b/bot/cogs/moderation.py index b596f36e6..5aa873a47 100644 --- a/bot/cogs/moderation.py +++ b/bot/cogs/moderation.py @@ -80,6 +80,43 @@ class Moderation(Scheduler, Cog):              if infraction["expires_at"] is not None:                  self.schedule_task(self.bot.loop, infraction["id"], infraction) +    @Cog.listener() +    async def on_member_join(self, member: Member) -> None: +        """Reapply active mute infractions for returning members.""" +        active_mutes = await self.bot.api_client.get( +            'bot/infractions', +            params={'user__id': str(member.id), 'type': 'mute', 'active': 'true'} +        ) +        if not active_mutes: +            return + +        # assume a single mute because of restrictions elsewhere +        mute = active_mutes[0] + +        # transform expiration to delay in seconds +        expiration_datetime = datetime.fromisoformat(mute["expires_at"][:-1]) +        delay = expiration_datetime - datetime.utcnow() +        delay_seconds = delay.total_seconds() + +        # if under a minute or in the past +        if delay_seconds < 60: +            log.debug(f"Marking infraction {mute['id']} as inactive (expired).") +            await self._deactivate_infraction(mute) +            self.cancel_task(mute["id"]) + +            # Notify the user that they've been unmuted. +            await self.notify_pardon( +                user=member, +                title="You have been unmuted.", +                content="You may now send messages in the server.", +                icon_url=Icons.user_unmute +            ) +            return + +        # allowing modlog since this is a passive action that should be logged +        await member.add_roles(self._muted_role, reason=f"Re-applying active mute: {mute['id']}") +        log.debug(f"User {member.id} has been re-muted on rejoin.") +      # region: Permanent infractions      @with_role(*MODERATION_ROLES) @@ -955,6 +992,11 @@ class Moderation(Scheduler, Cog):          user_id = infraction_object["user"]          infraction_type = infraction_object["type"] +        await self.bot.api_client.patch( +            'bot/infractions/' + str(infraction_object['id']), +            json={"active": False} +        ) +          if infraction_type == "mute":              member: Member = guild.get_member(user_id)              if member: @@ -970,11 +1012,6 @@ class Moderation(Scheduler, Cog):              except NotFound:                  log.info(f"Tried to unban user `{user_id}`, but Discord does not have an active ban registered.") -        await self.bot.api_client.patch( -            'bot/infractions/' + str(infraction_object['id']), -            json={"active": False} -        ) -      def _infraction_to_string(self, infraction_object: Dict[str, Union[str, int, bool]]) -> str:          """Convert the infraction object to a string representation."""          actor_id = infraction_object["actor"] diff --git a/bot/cogs/reddit.py b/bot/cogs/reddit.py index 63a57c5c6..6880aab85 100644 --- a/bot/cogs/reddit.py +++ b/bot/cogs/reddit.py @@ -21,6 +21,7 @@ class Reddit(Cog):      HEADERS = {"User-Agent": "Discord Bot: PythonDiscord (https://pythondiscord.com/)"}      URL = "https://www.reddit.com" +    MAX_FETCH_RETRIES = 3      def __init__(self, bot: Bot):          self.bot = bot @@ -42,16 +43,23 @@ class Reddit(Cog):          if params is None:              params = {} -        response = await self.bot.http_session.get( -            url=f"{self.URL}/{route}.json", -            headers=self.HEADERS, -            params=params -        ) +        url = f"{self.URL}/{route}.json" +        for _ in range(self.MAX_FETCH_RETRIES): +            response = await self.bot.http_session.get( +                url=url, +                headers=self.HEADERS, +                params=params +            ) +            if response.status == 200 and response.content_type == 'application/json': +                # Got appropriate response - process and return. +                content = await response.json() +                posts = content["data"]["children"] +                return posts[:amount] -        content = await response.json() -        posts = content["data"]["children"] +            await asyncio.sleep(3) -        return posts[:amount] +        log.debug(f"Invalid response from: {url} - status code {response.status}, mimetype {response.content_type}") +        return list()  # Failed to get appropriate response within allowed number of retries.      async def send_top_posts(          self, channel: TextChannel, subreddit: Subreddit, content: str = None, time: str = "all" @@ -62,13 +70,14 @@ class Reddit(Cog):          embed.description = ""          # Get the posts -        posts = await self.fetch_posts( -            route=f"{subreddit}/top", -            amount=5, -            params={ -                "t": time -            } -        ) +        async with channel.typing(): +            posts = await self.fetch_posts( +                route=f"{subreddit}/top", +                amount=5, +                params={ +                    "t": time +                } +            )          if not posts:              embed.title = random.choice(ERROR_REPLIES) diff --git a/bot/cogs/site.py b/bot/cogs/site.py index 4a423faa9..c3bdf85e4 100644 --- a/bot/cogs/site.py +++ b/bot/cogs/site.py @@ -44,17 +44,29 @@ class Site(Cog):      async def site_resources(self, ctx: Context) -> None:          """Info about the site's Resources page."""          learning_url = f"{PAGES_URL}/resources" -        tools_url = f"{PAGES_URL}/tools" -        embed = Embed(title="Resources & Tools") -        embed.set_footer(text=f"{learning_url} | {tools_url}") +        embed = Embed(title="Resources") +        embed.set_footer(text=f"{learning_url}")          embed.colour = Colour.blurple()          embed.description = (              f"The [Resources page]({learning_url}) on our website contains a " -            "list of hand-selected goodies that we regularly recommend " -            f"to both beginners and experts. The [Tools page]({tools_url}) " -            "contains a couple of the most popular tools for programming in " -            "Python." +            "list of hand-selected learning resources that we regularly recommend " +            f"to both beginners and experts." +        ) + +        await ctx.send(embed=embed) + +    @site_group.command(name="tools") +    async def site_tools(self, ctx: Context) -> None: +        """Info about the site's Tools page.""" +        tools_url = f"{PAGES_URL}/tools" + +        embed = Embed(title="Tools") +        embed.set_footer(text=f"{tools_url}") +        embed.colour = Colour.blurple() +        embed.description = ( +            f"The [Tools page]({tools_url}) on our website contains a " +            f"couple of the most popular tools for programming in Python."          )          await ctx.send(embed=embed) diff --git a/bot/cogs/tags.py b/bot/cogs/tags.py index b9dd3595e..cd70e783a 100644 --- a/bot/cogs/tags.py +++ b/bot/cogs/tags.py @@ -86,7 +86,7 @@ class Tags(Cog):                      max_lines=15                  ) -    @tags_group.command(name='set', aliases=('add', 'edit', 's')) +    @tags_group.command(name='set', aliases=('add', 's'))      @with_role(*MODERATION_ROLES)      async def set_command(          self, @@ -95,7 +95,7 @@ class Tags(Cog):          *,          tag_content: TagContentConverter,      ) -> None: -        """Create a new tag or update an existing one.""" +        """Create a new tag."""          body = {              'title': tag_name.lower().strip(),              'embed': { @@ -116,6 +116,35 @@ class Tags(Cog):              colour=Colour.blurple()          )) +    @tags_group.command(name='edit', aliases=('e', )) +    @with_role(*MODERATION_ROLES) +    async def edit_command( +        self, +        ctx: Context, +        tag_name: TagNameConverter, +        *, +        tag_content: TagContentConverter, +    ) -> None: +        """Edit an existing tag.""" +        body = { +            'embed': { +                'title': tag_name, +                'description': tag_content +            } +        } + +        await self.bot.api_client.patch(f'bot/tags/{tag_name}', json=body) + +        log.debug(f"{ctx.author} successfully edited the following tag in our database: \n" +                  f"tag_name: {tag_name}\n" +                  f"tag_content: '{tag_content}'\n") + +        await ctx.send(embed=Embed( +            title="Tag successfully edited", +            description=f"**{tag_name}** edited in the database.", +            colour=Colour.blurple() +        )) +      @tags_group.command(name='delete', aliases=('remove', 'rm', 'd'))      @with_role(Roles.admin, Roles.owner)      async def delete_command(self, ctx: Context, *, tag_name: TagNameConverter) -> None: diff --git a/bot/cogs/watchchannels/watchchannel.py b/bot/cogs/watchchannels/watchchannel.py index ce8014d69..760e012eb 100644 --- a/bot/cogs/watchchannels/watchchannel.py +++ b/bot/cogs/watchchannels/watchchannel.py @@ -335,7 +335,7 @@ class WatchChannel(metaclass=CogABCMeta):      def cog_unload(self) -> None:          """Takes care of unloading the cog and canceling the consumption task."""          self.log.trace(f"Unloading the cog") -        if not self._consume_task.done(): +        if self._consume_task and not self._consume_task.done():              self._consume_task.cancel()              try:                  self._consume_task.result() diff --git a/bot/converters.py b/bot/converters.py index 27223e632..cf0496541 100644 --- a/bot/converters.py +++ b/bot/converters.py @@ -51,7 +51,7 @@ class ValidURL(Converter):              async with ctx.bot.http_session.get(url) as resp:                  if resp.status != 200:                      raise BadArgument( -                        f"HTTP GET on `{url}` returned status `{resp.status_code}`, expected 200" +                        f"HTTP GET on `{url}` returned status `{resp.status}`, expected 200"                      )          except CertificateError:              if url.startswith('https'): | 
