| 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
 | import asyncio
import inspect
import itertools
from collections import namedtuple
from contextlib import suppress
from typing import Union
from discord import Colour, Embed, HTTPException, Message, Reaction, User
from discord.ext import commands
from discord.ext.commands import Bot, CheckFailure, Cog as DiscordCog, Command, Context
from fuzzywuzzy import fuzz, process
from bot import constants
from bot.constants import Channels, STAFF_ROLES
from bot.decorators import redirect_output
from bot.pagination import (
    DELETE_EMOJI, FIRST_EMOJI, LAST_EMOJI,
    LEFT_EMOJI, LinePaginator, RIGHT_EMOJI,
)
REACTIONS = {
    FIRST_EMOJI: 'first',
    LEFT_EMOJI: 'back',
    RIGHT_EMOJI: 'next',
    LAST_EMOJI: 'end',
    DELETE_EMOJI: 'stop'
}
Cog = namedtuple('Cog', ['name', 'description', 'commands'])
class HelpQueryNotFound(ValueError):
    """
    Raised when a HelpSession Query doesn't match a command or cog.
    Contains the custom attribute of ``possible_matches``.
    Attributes
    ----------
    possible_matches: dict
        Any commands that were close to matching the Query.
        The possible matched command names are the keys.
        The likeness match scores are the values.
    """
    def __init__(self, arg: str, possible_matches: dict = None):
        super().__init__(arg)
        self.possible_matches = possible_matches
class HelpSession:
    """
    An interactive session for bot and command help output.
    Attributes
    ----------
    title: str
        The title of the help message.
    query: Union[:class:`discord.ext.commands.Bot`,
                 :class:`discord.ext.commands.Command]
    description: str
        The description of the query.
    pages: list[str]
        A list of the help content split into manageable pages.
    message: :class:`discord.Message`
        The message object that's showing the help contents.
    destination: :class:`discord.abc.Messageable`
        Where the help message is to be sent to.
    """
    def __init__(
        self, ctx: Context, *command, cleanup: bool = False, only_can_run: bool = True,
        show_hidden: bool = False, max_lines: int = 15
    ):
        """
        Creates an instance of the HelpSession class.
        Parameters
        ----------
        ctx: :class:`discord.Context`
            The context of the invoked help command.
        *command: str
            A variable argument of the command being queried.
        cleanup: Optional[bool]
            Set to ``True`` to have the message deleted on timeout.
            If ``False``, it will clear all reactions on timeout.
            Defaults to ``False``.
        only_can_run: Optional[bool]
            Set to ``True`` to hide commands the user can't run.
            Defaults to ``False``.
        show_hidden: Optional[bool]
            Set to ``True`` to include hidden commands.
            Defaults to ``False``.
        max_lines: Optional[int]
            Sets the max number of lines the paginator will add to a
            single page.
            Defaults to 20.
        """
        self._ctx = ctx
        self._bot = ctx.bot
        self.title = "Command Help"
        # set the query details for the session
        if command:
            query_str = ' '.join(command)
            self.query = self._get_query(query_str)
            self.description = self.query.description or self.query.help
        else:
            self.query = ctx.bot
            self.description = self.query.description
        self.author = ctx.author
        self.destination = ctx.channel
        # set the config for the session
        self._cleanup = cleanup
        self._only_can_run = only_can_run
        self._show_hidden = show_hidden
        self._max_lines = max_lines
        # init session states
        self._pages = None
        self._current_page = 0
        self.message = None
        self._timeout_task = None
        self.reset_timeout()
    def _get_query(self, query: str) -> Union[Command, Cog]:
        """Attempts to match the provided query with a valid command or cog."""
        command = self._bot.get_command(query)
        if command:
            return command
        cog = self._bot.cogs.get(query)
        if cog:
            return Cog(
                name=cog.__class__.__name__,
                description=inspect.getdoc(cog),
                commands=[c for c in self._bot.commands if c.instance is cog]
            )
        self._handle_not_found(query)
    def _handle_not_found(self, query: str) -> None:
        """
        Handles when a query does not match a valid command or cog.
        Will pass on possible close matches along with the ``HelpQueryNotFound`` exception.
        Parameters
        ----------
        query: str
            The full query that was requested.
        Raises
        ------
        HelpQueryNotFound
        """
        # combine command and cog names
        choices = list(self._bot.all_commands) + list(self._bot.cogs)
        result = process.extractBests(query, choices, scorer=fuzz.ratio, score_cutoff=90)
        raise HelpQueryNotFound(f'Query "{query}" not found.', dict(result))
    async def timeout(self, seconds: int = 30) -> None:
        """
        Waits for a set number of seconds, then stops the help session.
        Parameters
        ----------
        seconds: int
            Number of seconds to wait.
        """
        await asyncio.sleep(seconds)
        await self.stop()
    def reset_timeout(self) -> None:
        """Cancels the original timeout task and sets it again from the start."""
        # cancel original if it exists
        if self._timeout_task:
            if not self._timeout_task.cancelled():
                self._timeout_task.cancel()
        # recreate the timeout task
        self._timeout_task = self._bot.loop.create_task(self.timeout())
    async def on_reaction_add(self, reaction: Reaction, user: User) -> None:
        """
        Event handler for when reactions are added on the help message.
        Parameters
        ----------
        reaction: :class:`discord.Reaction`
            The reaction that was added.
        user: :class:`discord.User`
            The user who added the reaction.
        """
        # ensure it was the relevant session message
        if reaction.message.id != self.message.id:
            return
        # ensure it was the session author who reacted
        if user.id != self.author.id:
            return
        emoji = str(reaction.emoji)
        # check if valid action
        if emoji not in REACTIONS:
            return
        self.reset_timeout()
        # Run relevant action method
        action = getattr(self, f'do_{REACTIONS[emoji]}', None)
        if action:
            await action()
        # remove the added reaction to prep for re-use
        with suppress(HTTPException):
            await self.message.remove_reaction(reaction, user)
    async def on_message_delete(self, message: Message) -> None:
        """Closes the help session when the help message is deleted."""
        if message.id == self.message.id:
            await self.stop()
    async def prepare(self) -> None:
        """Sets up the help session pages, events, message and reactions."""
        # create paginated content
        await self.build_pages()
        # setup listeners
        self._bot.add_listener(self.on_reaction_add)
        self._bot.add_listener(self.on_message_delete)
        # Send the help message
        await self.update_page()
        self.add_reactions()
    def add_reactions(self) -> None:
        """Adds the relevant reactions to the help message based on if pagination is required."""
        # if paginating
        if len(self._pages) > 1:
            for reaction in REACTIONS:
                self._bot.loop.create_task(self.message.add_reaction(reaction))
        # if single-page
        else:
            self._bot.loop.create_task(self.message.add_reaction(DELETE_EMOJI))
    def _category_key(self, cmd: Command) -> str:
        """
        Returns a cog name of a given command for use as a key for ``sorted`` and ``groupby``.
        A zero width space is used as a prefix for results with no cogs to force them last in ordering.
        """
        cog = cmd.cog_name
        return f'**{cog}**' if cog else f'**\u200bNo Category:**'
    def _get_command_params(self, cmd: Command) -> str:
        """
        Returns the command usage signature.
        This is a custom implementation of ``command.signature`` in order to format the command
        signature without aliases.
        """
        results = []
        for name, param in cmd.clean_params.items():
            # if argument has a default value
            if param.default is not param.empty:
                if isinstance(param.default, str):
                    show_default = param.default
                else:
                    show_default = param.default is not None
                # if default is not an empty string or None
                if show_default:
                    results.append(f'[{name}={param.default}]')
                else:
                    results.append(f'[{name}]')
            # if variable length argument
            elif param.kind == param.VAR_POSITIONAL:
                results.append(f'[{name}...]')
            # if required
            else:
                results.append(f'<{name}>')
        return f"{cmd.name} {' '.join(results)}"
    async def build_pages(self) -> None:
        """Builds the list of content pages to be paginated through in the help message, as a list of str."""
        # Use LinePaginator to restrict embed line height
        paginator = LinePaginator(prefix='', suffix='', max_lines=self._max_lines)
        prefix = constants.Bot.prefix
        # show signature if query is a command
        if isinstance(self.query, commands.Command):
            signature = self._get_command_params(self.query)
            parent = self.query.full_parent_name + ' ' if self.query.parent else ''
            paginator.add_line(f'**```{prefix}{parent}{signature}```**')
            # show command aliases
            aliases = ', '.join(f'`{a}`' for a in self.query.aliases)
            if aliases:
                paginator.add_line(f'**Can also use:** {aliases}\n')
            if not await self.query.can_run(self._ctx):
                paginator.add_line('***You cannot run this command.***\n')
        # show name if query is a cog
        if isinstance(self.query, Cog):
            paginator.add_line(f'**{self.query.name}**')
        if self.description:
            paginator.add_line(f'*{self.description}*')
        # list all children commands of the queried object
        if isinstance(self.query, (commands.GroupMixin, Cog)):
            # remove hidden commands if session is not wanting hiddens
            if not self._show_hidden:
                filtered = [c for c in self.query.commands if not c.hidden]
            else:
                filtered = self.query.commands
            # if after filter there are no commands, finish up
            if not filtered:
                self._pages = paginator.pages
                return
            # set category to Commands if cog
            if isinstance(self.query, Cog):
                grouped = (('**Commands:**', self.query.commands),)
            # set category to Subcommands if command
            elif isinstance(self.query, commands.Command):
                grouped = (('**Subcommands:**', self.query.commands),)
                # don't show prefix for subcommands
                prefix = ''
            # otherwise sort and organise all commands into categories
            else:
                cat_sort = sorted(filtered, key=self._category_key)
                grouped = itertools.groupby(cat_sort, key=self._category_key)
            # process each category
            for category, cmds in grouped:
                cmds = sorted(cmds, key=lambda c: c.name)
                # if there are no commands, skip category
                if len(cmds) == 0:
                    continue
                cat_cmds = []
                # format details for each child command
                for command in cmds:
                    # skip if hidden and hide if session is set to
                    if command.hidden and not self._show_hidden:
                        continue
                    # see if the user can run the command
                    strikeout = ''
                    # Patch to make the !help command work outside of #bot-commands again
                    # This probably needs a proper rewrite, but this will make it work in
                    # the mean time.
                    try:
                        can_run = await command.can_run(self._ctx)
                    except CheckFailure:
                        can_run = False
                    if not can_run:
                        # skip if we don't show commands they can't run
                        if self._only_can_run:
                            continue
                        strikeout = '~~'
                    signature = self._get_command_params(command)
                    info = f"{strikeout}**`{prefix}{signature}`**{strikeout}"
                    # handle if the command has no docstring
                    if command.short_doc:
                        cat_cmds.append(f'{info}\n*{command.short_doc}*')
                    else:
                        cat_cmds.append(f'{info}\n*No details provided.*')
                # state var for if the category should be added next
                print_cat = 1
                new_page = True
                for details in cat_cmds:
                    # keep details together, paginating early if it won't fit
                    lines_adding = len(details.split('\n')) + print_cat
                    if paginator._linecount + lines_adding > self._max_lines:
                        paginator._linecount = 0
                        new_page = True
                        paginator.close_page()
                        # new page so print category title again
                        print_cat = 1
                    if print_cat:
                        if new_page:
                            paginator.add_line('')
                        paginator.add_line(category)
                        print_cat = 0
                    paginator.add_line(details)
        # save organised pages to session
        self._pages = paginator.pages
    def embed_page(self, page_number: int = 0) -> Embed:
        """Returns an Embed with the requested page formatted within."""
        embed = Embed()
        # if command or cog, add query to title for pages other than first
        if isinstance(self.query, (commands.Command, Cog)) and page_number > 0:
            title = f'Command Help | "{self.query.name}"'
        else:
            title = self.title
        embed.set_author(name=title, icon_url=constants.Icons.questionmark)
        embed.description = self._pages[page_number]
        # add page counter to footer if paginating
        page_count = len(self._pages)
        if page_count > 1:
            embed.set_footer(text=f'Page {self._current_page+1} / {page_count}')
        return embed
    async def update_page(self, page_number: int = 0) -> None:
        """Sends the intial message, or changes the existing one to the given page number."""
        self._current_page = page_number
        embed_page = self.embed_page(page_number)
        if not self.message:
            self.message = await self.destination.send(embed=embed_page)
        else:
            await self.message.edit(embed=embed_page)
    @classmethod
    async def start(cls, ctx: Context, *command, **options) -> "HelpSession":
        """
        Create and begin a help session based on the given command context.
        Parameters
        ----------
        ctx: :class:`discord.ext.commands.Context`
        The context of the invoked help command.
        *command: str
            A variable argument of the command being queried.
        cleanup: Optional[bool]
            Set to ``True`` to have the message deleted on session end.
            Defaults to ``False``.
        only_can_run: Optional[bool]
            Set to ``True`` to hide commands the user can't run.
            Defaults to ``False``.
        show_hidden: Optional[bool]
            Set to ``True`` to include hidden commands.
            Defaults to ``False``.
        max_lines: Optional[int]
            Sets the max number of lines the paginator will add to a
            single page.
            Defaults to 20.
        """
        session = cls(ctx, *command, **options)
        await session.prepare()
        return session
    async def stop(self) -> None:
        """Stops the help session, removes event listeners and attempts to delete the help message."""
        self._bot.remove_listener(self.on_reaction_add)
        self._bot.remove_listener(self.on_message_delete)
        # ignore if permission issue, or the message doesn't exist
        with suppress(HTTPException, AttributeError):
            if self._cleanup:
                await self.message.delete()
            else:
                await self.message.clear_reactions()
    @property
    def is_first_page(self) -> bool:
        """Check if session is currently showing the first page."""
        return self._current_page == 0
    @property
    def is_last_page(self) -> bool:
        """Check if the session is currently showing the last page."""
        return self._current_page == (len(self._pages)-1)
    async def do_first(self) -> None:
        """Event that is called when the user requests the first page."""
        if not self.is_first_page:
            await self.update_page(0)
    async def do_back(self) -> None:
        """Event that is called when the user requests the previous page."""
        if not self.is_first_page:
            await self.update_page(self._current_page-1)
    async def do_next(self) -> None:
        """Event that is called when the user requests the next page."""
        if not self.is_last_page:
            await self.update_page(self._current_page+1)
    async def do_end(self) -> None:
        """Event that is called when the user requests the last page."""
        if not self.is_last_page:
            await self.update_page(len(self._pages)-1)
    async def do_stop(self) -> None:
        """Event that is called when the user requests to stop the help session."""
        await self.message.delete()
class Help(DiscordCog):
    """Custom Embed Pagination Help feature."""
    @commands.command('help')
    @redirect_output(destination_channel=Channels.bot, bypass_roles=STAFF_ROLES)
    async def new_help(self, ctx: Context, *commands) -> None:
        """Shows Command Help."""
        try:
            await HelpSession.start(ctx, *commands)
        except HelpQueryNotFound as error:
            embed = Embed()
            embed.colour = Colour.red()
            embed.title = str(error)
            if error.possible_matches:
                matches = '\n'.join(error.possible_matches.keys())
                embed.description = f'**Did you mean:**\n`{matches}`'
            await ctx.send(embed=embed)
def unload(bot: Bot) -> None:
    """
    Reinstates the original help command.
    This is run if the cog raises an exception on load, or if the extension is unloaded.
    """
    bot.remove_command('help')
    bot.add_command(bot._old_help)
def setup(bot: Bot) -> None:
    """
    The setup for the help extension.
    This is called automatically on `bot.load_extension` being run.
    Stores the original help command instance on the ``bot._old_help``
    attribute for later reinstatement, before removing it from the
    command registry so the new help command can be loaded successfully.
    If an exception is raised during the loading of the cog, ``unload``
    will be called in order to reinstate the original help command.
    """
    bot._old_help = bot.get_command('help')
    bot.remove_command('help')
    try:
        bot.add_cog(Help())
    except Exception:
        unload(bot)
        raise
def teardown(bot: Bot) -> None:
    """
    The teardown for the help extension.
    This is called automatically on `bot.unload_extension` being run.
    Calls ``unload`` in order to reinstate the original help command.
    """
    unload(bot)
 |