aboutsummaryrefslogtreecommitdiffstats
path: root/bot/exts/utils/reminders.py
blob: 56df259238b518f60c7afe002f9c313442084d43 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
import random
import textwrap
import typing as t
from datetime import UTC, datetime
from operator import itemgetter

import discord
from dateutil.parser import isoparse
from discord import Interaction
from discord.ext.commands import Cog, Context, Greedy, group
from pydis_core.site_api import ResponseCodeError
from pydis_core.utils import scheduling
from pydis_core.utils.members import get_or_fetch_member
from pydis_core.utils.scheduling import Scheduler

from bot.bot import Bot
from bot.constants import (
    Channels,
    Guild,
    Icons,
    MODERATION_ROLES,
    NEGATIVE_REPLIES,
    POSITIVE_REPLIES,
    Roles,
    STAFF_PARTNERS_COMMUNITY_ROLES,
)
from bot.converters import Duration, UnambiguousUser
from bot.errors import LockedResourceError
from bot.log import get_logger
from bot.pagination import LinePaginator
from bot.utils import time
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

log = get_logger(__name__)

LOCK_NAMESPACE = "reminder"
WHITELISTED_CHANNELS = Guild.reminder_whitelist
MAXIMUM_REMINDERS = 5
REMINDER_EDIT_CONFIRMATION_TIMEOUT = 60

Mentionable = discord.Member | discord.Role
ReminderMention = UnambiguousUser | discord.Role


class ModifyReminderConfirmationView(discord.ui.View):
    """A view to confirm modifying someone else's reminder by admins."""

    def __init__(self, author: discord.Member | discord.User):
        super().__init__(timeout=CONFIRMATION_TIMEOUT)
        self.author = author
        self.result: bool | None = None

    async def interaction_check(self, interaction: Interaction) -> bool:
        """Only allow interactions from the command invoker."""
        return interaction.user.id == self.author.id

    async def on_timeout(self) -> None:
        """Default to not modifying if the user doesn't respond."""
        self.result = False

    @discord.ui.button(label="Confirm", style=discord.ButtonStyle.blurple, row=0)
    async def confirm(self, interaction: Interaction, button: discord.ui.Button) -> None:
        """Confirm the reminder modification."""
        await interaction.response.edit_message(view=None)
        self.result = True
        self.stop()

    @discord.ui.button(label="Cancel", row=0)
    async def cancel(self, interaction: Interaction, button: discord.ui.Button) -> None:
        """Cancel the reminder modification."""
        await interaction.response.edit_message(content="🚫 Operation canceled.", view=None)
        self.result = False
        self.stop()


class Reminders(Cog):
    """Provide in-channel reminder functionality."""

    def __init__(self, bot: Bot):
        self.bot = bot
        self.scheduler = Scheduler(self.__class__.__name__)

    async def cog_unload(self) -> None:
        """Cancel scheduled tasks."""
        self.scheduler.cancel_all()

    async def cog_load(self) -> None:
        """Get all current reminders from the API and reschedule them."""
        await self.bot.wait_until_guild_available()
        response = await self.bot.api_client.get(
            "bot/reminders",
            params={"active": "true"}
        )

        now = datetime.now(UTC)

        for reminder in response:
            is_valid, *_ = self.ensure_valid_reminder(reminder)
            if not is_valid:
                continue

            remind_at = isoparse(reminder["expiration"])

            # If the reminder is already overdue ...
            if remind_at < now:
                await self.send_reminder(reminder, remind_at)
            else:
                self.schedule_reminder(reminder)

    def ensure_valid_reminder(self, reminder: dict) -> tuple[bool, discord.TextChannel]:
        """Ensure reminder channel can be fetched otherwise delete the reminder."""
        channel = self.bot.get_channel(reminder["channel_id"])
        is_valid = True
        if not channel:
            is_valid = False
            log.info(
                f"Reminder {reminder['id']} invalid: "
                f"Channel {reminder['channel_id']}={channel}."
            )
            scheduling.create_task(self.bot.api_client.delete(f"bot/reminders/{reminder['id']}"))

        return is_valid, channel

    @staticmethod
    async def _send_confirmation(
        ctx: Context,
        on_success: str,
        reminder_id: str | int
    ) -> None:
        """Send an embed confirming the reminder change was made successfully."""
        embed = discord.Embed(
            description=on_success,
            colour=discord.Colour.green(),
            title=random.choice(POSITIVE_REPLIES)
        )

        footer_str = f"ID: {reminder_id}"

        embed.set_footer(text=footer_str)

        await ctx.send(embed=embed)

    @staticmethod
    async def _check_mentions(ctx: Context, mentions: t.Iterable[Mentionable]) -> tuple[bool, str]:
        """
        Returns whether or not the list of mentions is allowed.

        Conditions:
        - Role reminders are Mods+
        - Reminders for other users are Helpers+

        If mentions aren't allowed, also return the type of mention(s) disallowed.
        """
        if await has_no_roles_check(ctx, *STAFF_PARTNERS_COMMUNITY_ROLES):
            return False, "members/roles"
        if await has_no_roles_check(ctx, *MODERATION_ROLES):
            return all(isinstance(mention, discord.User | discord.Member) for mention in mentions), "roles"
        return True, ""

    @staticmethod
    async def validate_mentions(ctx: Context, mentions: t.Iterable[Mentionable]) -> bool:
        """
        Filter mentions to see if the user can mention, and sends a denial if not allowed.

        Returns whether or not the validation is successful.
        """
        mentions_allowed, disallowed_mentions = await Reminders._check_mentions(ctx, mentions)

        if not mentions or mentions_allowed:
            return True
        await send_denial(ctx, f"You can't mention other {disallowed_mentions} in your reminder!")
        return False

    async def get_mentionables(self, mention_ids: list[int]) -> t.Iterator[Mentionable]:
        """Converts Role and Member ids to their corresponding objects if possible."""
        guild = self.bot.get_guild(Guild.id)
        for mention_id in mention_ids:
            member = await get_or_fetch_member(guild, mention_id)
            if mentionable := (member or guild.get_role(mention_id)):
                yield mentionable

    def schedule_reminder(self, reminder: dict) -> None:
        """A coroutine which sends the reminder once the time is reached, and cancels the running task."""
        reminder_datetime = isoparse(reminder["expiration"])
        self.scheduler.schedule_at(reminder_datetime, reminder["id"], self.send_reminder(reminder))

    async def _edit_reminder(self, reminder_id: int, payload: dict) -> dict:
        """
        Edits a reminder in the database given the ID and payload.

        Returns the edited reminder.
        """
        # Send the request to update the reminder in the database
        reminder = await self.bot.api_client.patch(
            "bot/reminders/" + str(reminder_id),
            json=payload
        )
        return reminder

    async def _reschedule_reminder(self, reminder: dict) -> None:
        """Reschedule a reminder object."""
        log.trace(f"Cancelling old task #{reminder['id']}")
        self.scheduler.cancel(reminder["id"])

        log.trace(f"Scheduling new task #{reminder['id']}")
        self.schedule_reminder(reminder)

    @lock_arg(LOCK_NAMESPACE, "reminder", itemgetter("id"), raise_error=True)
    async def send_reminder(self, reminder: dict, expected_time: time.Timestamp | None = None) -> None:
        """Send the reminder."""
        is_valid, 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()
        if expected_time:
            embed.colour = discord.Colour.red()
            embed.set_author(
                icon_url=Icons.remind_red,
                name="Sorry, your reminder should have arrived earlier!"
            )
        else:
            embed.colour = discord.Colour.og_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 async for mentionable in self.get_mentionables(reminder["mentions"])
        ])

        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"<@{reminder['author']}> {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']}")

    @staticmethod
    async def try_get_content_from_reply(ctx: Context) -> str | None:
        """
        Attempts to get content from the referenced message, if applicable.

        Differs from pydis_core.utils.commands.clean_text_or_reply as allows for messages with no content.
        """
        content = None
        if reference := ctx.message.reference:
            if isinstance((resolved_message := reference.resolved), discord.Message):
                content = resolved_message.content

        # If we weren't able to get the content of a replied message
        if content is None:
            await send_denial(ctx, "Your reminder must have a content and/or reply to a message.")
            return None

        # If the replied message has no content (e.g. only attachments/embeds)
        if content == "":
            content = "*See referenced message.*"

        return content

    @group(name="remind", aliases=("reminder", "reminders", "remindme"), invoke_without_command=True)
    async def remind_group(
        self, ctx: Context, mentions: Greedy[ReminderMention], expiration: Duration, *, content: str | None = None
    ) -> None:
        """
        Commands for managing your reminders.

        The `expiration` duration of `!remind new` supports the following symbols for each unit of time:
        - years: `Y`, `y`, `year`, `years`
        - months: `m`, `month`, `months`
        - weeks: `w`, `W`, `week`, `weeks`
        - days: `d`, `D`, `day`, `days`
        - hours: `H`, `h`, `hour`, `hours`
        - minutes: `M`, `minute`, `minutes`
        - seconds: `S`, `s`, `second`, `seconds`

        For example, to set a reminder that expires in 3 days and 1 minute, you can do `!remind new 3d1M Do something`.
        """
        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[ReminderMention], expiration: Duration, *, content: str | None = None
    ) -> None:
        """
        Set yourself a simple reminder.

        The `expiration` duration supports the following symbols for each unit of time:
        - years: `Y`, `y`, `year`, `years`
        - months: `m`, `month`, `months`
        - weeks: `w`, `W`, `week`, `weeks`
        - days: `d`, `D`, `day`, `days`
        - hours: `H`, `h`, `hour`, `hours`
        - minutes: `M`, `minute`, `minutes`
        - seconds: `S`, `s`, `second`, `seconds`

        For example, to set a reminder that expires in 3 days and 1 minute, you can do `!remind new 3d1M Do something`.
        """
        # If the user is not staff, partner or part of the python community,
        # we need to verify whether or not to make a reminder at all.
        if await has_no_roles_check(ctx, *STAFF_PARTNERS_COMMUNITY_ROLES):

            # If they don't have permission to set a reminder in this channel
            if ctx.channel.id not in WHITELISTED_CHANNELS:
                bot_commands = ctx.guild.get_channel(Channels.bot_commands)
                await send_denial(ctx, f"Sorry, you can only do that in {bot_commands.mention}!")
                return

            # Get their current active reminders
            active_reminders = await self.bot.api_client.get(
                "bot/reminders",
                params={
                    "author__id": str(ctx.author.id)
                }
            )

            # Let's limit this, so we don't get 10 000
            # reminders from kip or something like that :P
            if len(active_reminders) > MAXIMUM_REMINDERS:
                await send_denial(ctx, "You have too many active reminders!")
                return

        # Remove duplicate mentions
        mentions = set(mentions)
        mentions.discard(ctx.author)

        # Filter mentions to see if the user can mention members/roles
        if not await self.validate_mentions(ctx, mentions):
            return

        mention_ids = [mention.id for mention in mentions]

        # If `content` isn't provided then we try to get message content of a replied message
        if not content:
            content = await self.try_get_content_from_reply(ctx)
            if not content:
                # Couldn't get content from reply
                return

        # Now we can attempt to actually set the reminder.
        reminder = await self.bot.api_client.post(
            "bot/reminders",
            json={
                "author": ctx.author.id,
                "channel_id": ctx.message.channel.id,
                "jump_url": ctx.message.jump_url,
                "content": content,
                "expiration": expiration.isoformat(),
                "mentions": mention_ids,
            }
        )

        formatted_time = time.discord_timestamp(expiration, time.TimestampFormats.DAY_TIME)
        mention_string = f"Your reminder will arrive on {formatted_time}"

        if mentions:
            mention_string += f" and will mention {len(mentions)} other(s)"
        mention_string += "!"

        # Confirm to the user that it worked.
        await self._send_confirmation(
            ctx,
            on_success=mention_string,
            reminder_id=reminder["id"]
        )

        self.schedule_reminder(reminder)

    @remind_group.command(name="list")
    async def list_reminders(self, ctx: Context) -> None:
        """View a paginated embed of all reminders for your user."""
        # Get all the user's reminders from the database.
        data = await self.bot.api_client.get(
            "bot/reminders",
            params={"author__id": str(ctx.author.id)}
        )

        # Make a list of tuples so it can be sorted by time.
        reminders = sorted(
            (
                (rem["content"], rem["expiration"], rem["id"], rem["mentions"])
                for rem in data
            ),
            key=itemgetter(1)
        )

        lines = []

        for content, remind_at, id_, mentions in reminders:
            # Parse and humanize the time, make it pretty :D
            expiry = time.format_relative(remind_at)

            mentions = ", ".join([
                # Both Role and User objects have the `mention` attribute
                f"{mentionable.mention} ({mentionable})" async for mentionable in self.get_mentionables(mentions)
            ])
            mention_string = f"\n**Mentions:** {mentions}" if mentions else ""

            text = textwrap.dedent(f"""
            **Reminder #{id_}:** *expires {expiry}* (ID: {id_}){mention_string}
            {content}
            """).strip()

            lines.append(text)

        embed = discord.Embed()
        embed.colour = discord.Colour.og_blurple()
        embed.title = f"Reminders for {ctx.author}"

        # Remind the user that they have no reminders :^)
        if not lines:
            embed.description = "No active reminders could be found."
            await ctx.send(embed=embed)
            return

        # Construct the embed and paginate it.
        embed.colour = discord.Colour.og_blurple()

        await LinePaginator.paginate(
            lines,
            ctx, embed,
            max_lines=3,
        )

    @remind_group.group(name="edit", aliases=("change", "modify"), invoke_without_command=True)
    async def edit_reminder_group(self, ctx: Context) -> None:
        """Commands for modifying your current reminders."""
        await ctx.send_help(ctx.command)

    @edit_reminder_group.command(name="duration", aliases=("time",))
    async def edit_reminder_duration(self, ctx: Context, id_: int, expiration: Duration) -> None:
        """
        Edit one of your reminder's expiration.

        The `expiration` duration supports the following symbols for each unit of time:
        - years: `Y`, `y`, `year`, `years`
        - months: `m`, `month`, `months`
        - weeks: `w`, `W`, `week`, `weeks`
        - days: `d`, `D`, `day`, `days`
        - hours: `H`, `h`, `hour`, `hours`
        - minutes: `M`, `minute`, `minutes`
        - seconds: `S`, `s`, `second`, `seconds`

        For example, to edit a reminder to expire in 3 days and 1 minute, you can do `!remind edit duration 1234 3d1M`.
        """
        await self.edit_reminder(ctx, id_, {"expiration": expiration.isoformat()})

    @edit_reminder_group.command(name="content", aliases=("reason",))
    async def edit_reminder_content(self, ctx: Context, id_: int, *, content: str | None = None) -> None:
        """
        Edit one of your reminder's content.

        You can either supply the new content yourself, or reply to a message to use its content.
        """
        if not content:
            content = await self.try_get_content_from_reply(ctx)
            if not content:
                # Message doesn't have a reply to get content from
                return
        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[ReminderMention]) -> None:
        """Edit one of your reminder's mentions."""
        # Remove duplicate mentions
        mentions = set(mentions)
        mentions.discard(ctx.author)

        # Filter mentions to see if the user can mention members/roles
        if not await self.validate_mentions(ctx, mentions):
            return

        mention_ids = [mention.id for mention in mentions]
        await self.edit_reminder(ctx, id_, {"mentions": mention_ids})

    @lock_arg(LOCK_NAMESPACE, "id_", raise_error=True)
    async def edit_reminder(self, ctx: Context, id_: int, payload: dict) -> None:
        """Edits a reminder with the given payload, then sends a confirmation message."""
        if not await self._can_modify(ctx, id_):
            return
        reminder = await self._edit_reminder(id_, payload)

        # Send a confirmation message to the channel
        await self._send_confirmation(
            ctx,
            on_success="That reminder has been edited successfully!",
            reminder_id=id_,
        )
        await self._reschedule_reminder(reminder)

    @lock_arg(LOCK_NAMESPACE, "id_", raise_error=True)
    async def _delete_reminder(self, ctx: Context, id_: int) -> bool:
        """Acquires a lock on `id_` and returns `True` if reminder is deleted, otherwise `False`."""
        if not await self._can_modify(ctx, id_, send_on_denial=False):
            return False

        await self.bot.api_client.delete(f"bot/reminders/{id_}")
        self.scheduler.cancel(id_)
        return True

    @remind_group.command("delete", aliases=("remove", "cancel"))
    async def delete_reminder(self, ctx: Context, ids: Greedy[int]) -> None:
        """Delete up to (and including) 5 of your active reminders."""
        if len(ids) > 5:
            await send_denial(ctx, "You can only delete a maximum of 5 reminders at once.")
            return

        deleted_ids = []
        for id_ in set(ids):
            try:
                reminder_deleted = await self._delete_reminder(ctx, id_)
            except LockedResourceError:
                continue
            else:
                if reminder_deleted:
                    deleted_ids.append(str(id_))

        if deleted_ids:
            colour = discord.Colour.green()
            title = random.choice(POSITIVE_REPLIES)
            deletion_message = f"Successfully deleted the following reminder(s): {', '.join(deleted_ids)}"

            if len(deleted_ids) != len(ids):
                deletion_message += (
                    "\n\nThe other reminder(s) could not be deleted as they're either locked, "
                    "belong to someone else, or don't exist."
                )
        else:
            colour = discord.Colour.red()
            title = random.choice(NEGATIVE_REPLIES)
            deletion_message = (
                "Could not delete the reminder(s) as they're either locked, "
                "belong to someone else, or don't exist."
            )

        embed = discord.Embed(
            description=deletion_message,
            colour=colour,
            title=title
        )
        await ctx.send(embed=embed)

    async def _can_modify(self, ctx: Context, reminder_id: str | int, send_on_denial: bool = True) -> bool:
        """
        Check whether the reminder can be modified by the ctx author.

        The check passes if the user created the reminder, or if they are an admin (with confirmation).
        """
        try:
            api_response = await self.bot.api_client.get(f"bot/reminders/{reminder_id}")
        except ResponseCodeError as e:
            # Override error-handling so that a 404 message isn't sent to Discord when `send_on_denial` is `False`
            if not send_on_denial:
                if e.status == 404:
                    return False
            raise e

        if api_response["author"] == ctx.author.id:
            log.debug(f"{ctx.author} is the reminder's author and passes the check.")
            return True

        if await has_any_role_check(ctx, Roles.admins):
            log.debug(f"{ctx.author} is an admin, asking for confirmation to modify someone else's.")

            confirmation_view = ModifyConfirmationView(ctx.author)
            await ctx.send(
                "Are you sure you want to modify someone else's reminder?",
                view=confirmation_view,
            )
            await confirmation_view.wait()

            if confirmation_view.result:
                log.debug(f"{ctx.author} has confirmed reminder modification.")
            else:
                log.debug(f"{ctx.author} has cancelled reminder modification.")
            return confirmation_view.result

        log.debug(f"{ctx.author} is not the reminder author and does not pass the check.")
        if send_on_denial:
            await send_denial(ctx, "You can't modify reminders of other users!")
        return False


async def setup(bot: Bot) -> None:
    """Load the Reminders cog."""
    await bot.add_cog(Reminders(bot))