File: commands.rst

package info (click to toggle)
python-discord 2.5.2%2Bdfsg-1
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 8,180 kB
  • sloc: python: 46,013; javascript: 363; makefile: 154
file content (1274 lines) | stat: -rw-r--r-- 47,142 bytes parent folder | download | duplicates (2)
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
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
1001
1002
1003
1004
1005
1006
1007
1008
1009
1010
1011
1012
1013
1014
1015
1016
1017
1018
1019
1020
1021
1022
1023
1024
1025
1026
1027
1028
1029
1030
1031
1032
1033
1034
1035
1036
1037
1038
1039
1040
1041
1042
1043
1044
1045
1046
1047
1048
1049
1050
1051
1052
1053
1054
1055
1056
1057
1058
1059
1060
1061
1062
1063
1064
1065
1066
1067
1068
1069
1070
1071
1072
1073
1074
1075
1076
1077
1078
1079
1080
1081
1082
1083
1084
1085
1086
1087
1088
1089
1090
1091
1092
1093
1094
1095
1096
1097
1098
1099
1100
1101
1102
1103
1104
1105
1106
1107
1108
1109
1110
1111
1112
1113
1114
1115
1116
1117
1118
1119
1120
1121
1122
1123
1124
1125
1126
1127
1128
1129
1130
1131
1132
1133
1134
1135
1136
1137
1138
1139
1140
1141
1142
1143
1144
1145
1146
1147
1148
1149
1150
1151
1152
1153
1154
1155
1156
1157
1158
1159
1160
1161
1162
1163
1164
1165
1166
1167
1168
1169
1170
1171
1172
1173
1174
1175
1176
1177
1178
1179
1180
1181
1182
1183
1184
1185
1186
1187
1188
1189
1190
1191
1192
1193
1194
1195
1196
1197
1198
1199
1200
1201
1202
1203
1204
1205
1206
1207
1208
1209
1210
1211
1212
1213
1214
1215
1216
1217
1218
1219
1220
1221
1222
1223
1224
1225
1226
1227
1228
1229
1230
1231
1232
1233
1234
1235
1236
1237
1238
1239
1240
1241
1242
1243
1244
1245
1246
1247
1248
1249
1250
1251
1252
1253
1254
1255
1256
1257
1258
1259
1260
1261
1262
1263
1264
1265
1266
1267
1268
1269
1270
1271
1272
1273
1274
.. currentmodule:: discord

.. _ext_commands_commands:

Commands
==========

One of the most appealing aspects of the command extension is how easy it is to define commands and
how you can arbitrarily nest groups and commands to have a rich sub-command system.

Commands are defined by attaching it to a regular Python function. The command is then invoked by the user using a similar
signature to the Python function.

.. warning::

    You must have access to the :attr:`~discord.Intents.message_content` intent for the commands extension
    to function. This must be set both in the developer portal and within your code.

    Failure to do this will result in your bot not responding to any of your commands.

For example, in the given command definition:

.. code-block:: python3

    @bot.command()
    async def foo(ctx, arg):
        await ctx.send(arg)

With the following prefix (``$``), it would be invoked by the user via:

.. code-block:: none

    $foo abc

A command must always have at least one parameter, ``ctx``, which is the :class:`.Context` as the first one.

There are two ways of registering a command. The first one is by using :meth:`.Bot.command` decorator,
as seen in the example above. The second is using the :func:`~ext.commands.command` decorator followed by
:meth:`.Bot.add_command` on the instance.

Essentially, these two are equivalent: ::

    import discord
    from discord.ext import commands

    intents = discord.Intents.default()
    intents.message_content = True

    bot = commands.Bot(command_prefix='$', intents=intents)

    @bot.command()
    async def test(ctx):
        pass

    # or:

    @commands.command()
    async def test(ctx):
        pass

    bot.add_command(test)

Since the :meth:`.Bot.command` decorator is shorter and easier to comprehend, it will be the one used throughout the
documentation here.

Any parameter that is accepted by the :class:`~discord.ext.commands.Command` constructor can be passed into the decorator. For example, to change
the name to something other than the function would be as simple as doing this:

.. code-block:: python3

    @bot.command(name='list')
    async def _list(ctx, arg):
        pass

Parameters
------------

Since we define commands by making Python functions, we also define the argument passing behaviour by the function
parameters.

Certain parameter types do different things in the user side and most forms of parameter types are supported.

Positional
++++++++++++

The most basic form of parameter passing is the positional parameter. This is where we pass a parameter as-is:

.. code-block:: python3

    @bot.command()
    async def test(ctx, arg):
        await ctx.send(arg)


On the bot using side, you can provide positional arguments by just passing a regular string:

.. image:: /images/commands/positional1.png

To make use of a word with spaces in between, you should quote it:

.. image:: /images/commands/positional2.png

As a note of warning, if you omit the quotes, you will only get the first word:

.. image:: /images/commands/positional3.png

Since positional arguments are just regular Python arguments, you can have as many as you want:

.. code-block:: python3

    @bot.command()
    async def test(ctx, arg1, arg2):
        await ctx.send(f'You passed {arg1} and {arg2}')

Variable
++++++++++

Sometimes you want users to pass in an undetermined number of parameters. The library supports this
similar to how variable list parameters are done in Python:

.. code-block:: python3

    @bot.command()
    async def test(ctx, *args):
        arguments = ', '.join(args)
        await ctx.send(f'{len(args)} arguments: {arguments}')

This allows our user to accept either one or many arguments as they please. This works similar to positional arguments,
so multi-word parameters should be quoted.

For example, on the bot side:

.. image:: /images/commands/variable1.png

If the user wants to input a multi-word argument, they have to quote it like earlier:

.. image:: /images/commands/variable2.png

Do note that similar to the Python function behaviour, a user can technically pass no arguments
at all:

.. image:: /images/commands/variable3.png

Since the ``args`` variable is a :class:`py:tuple`,
you can do anything you would usually do with one.

Keyword-Only Arguments
++++++++++++++++++++++++

When you want to handle parsing of the argument yourself or do not feel like you want to wrap multi-word user input into
quotes, you can ask the library to give you the rest as a single argument. We do this by using a **keyword-only argument**,
seen below:

.. code-block:: python3

    @bot.command()
    async def test(ctx, *, arg):
        await ctx.send(arg)

.. warning::

    You can only have one keyword-only argument due to parsing ambiguities.

On the bot side, we do not need to quote input with spaces:

.. image:: /images/commands/keyword1.png

Do keep in mind that wrapping it in quotes leaves it as-is:

.. image:: /images/commands/keyword2.png

By default, the keyword-only arguments are stripped of white space to make it easier to work with. This behaviour can be
toggled by the :attr:`.Command.rest_is_raw` argument in the decorator.

.. _ext_commands_context:

Invocation Context
-------------------

As seen earlier, every command must take at least a single parameter, called the :class:`~ext.commands.Context`.

This parameter gives you access to something called the "invocation context". Essentially all the information you need to
know how the command was executed. It contains a lot of useful information:

- :attr:`.Context.guild` returns the :class:`Guild` of the command, if any.
- :attr:`.Context.message` returns the :class:`Message` of the command.
- :attr:`.Context.author` returns the :class:`Member` or :class:`User` that called the command.
- :meth:`.Context.send` to send a message to the channel the command was used in.

The context implements the :class:`abc.Messageable` interface, so anything you can do on a :class:`abc.Messageable` you
can do on the :class:`~ext.commands.Context`.

Converters
------------

Adding bot arguments with function parameters is only the first step in defining your bot's command interface. To actually
make use of the arguments, we usually want to convert the data into a target type. We call these
:ref:`ext_commands_api_converters`.

Converters come in a few flavours:

- A regular callable object that takes an argument as a sole parameter and returns a different type.

    - These range from your own function, to something like :class:`bool` or :class:`int`.

- A custom class that inherits from :class:`~ext.commands.Converter`.

.. _ext_commands_basic_converters:

Basic Converters
++++++++++++++++++

At its core, a basic converter is a callable that takes in an argument and turns it into something else.

For example, if we wanted to add two numbers together, we could request that they are turned into integers
for us by specifying the converter:

.. code-block:: python3

    @bot.command()
    async def add(ctx, a: int, b: int):
        await ctx.send(a + b)

We specify converters by using something called a **function annotation**. This is a Python 3 exclusive feature that was
introduced in :pep:`3107`.

This works with any callable, such as a function that would convert a string to all upper-case:

.. code-block:: python3

    def to_upper(argument):
        return argument.upper()

    @bot.command()
    async def up(ctx, *, content: to_upper):
        await ctx.send(content)

bool
^^^^^^

Unlike the other basic converters, the :class:`bool` converter is treated slightly different. Instead of casting directly to the :class:`bool` type, which would result in any non-empty argument returning ``True``, it instead evaluates the argument as ``True`` or ``False`` based on its given content:

.. code-block:: python3

    if lowered in ('yes', 'y', 'true', 't', '1', 'enable', 'on'):
        return True
    elif lowered in ('no', 'n', 'false', 'f', '0', 'disable', 'off'):
        return False

.. _ext_commands_adv_converters:

Advanced Converters
+++++++++++++++++++++

Sometimes a basic converter doesn't have enough information that we need. For example, sometimes we want to get some
information from the :class:`Message` that called the command or we want to do some asynchronous processing.

For this, the library provides the :class:`~ext.commands.Converter` interface. This allows you to have access to the
:class:`.Context` and have the callable be asynchronous. Defining a custom converter using this interface requires
overriding a single method, :meth:`.Converter.convert`.

An example converter:

.. code-block:: python3

    import random

    class Slapper(commands.Converter):
        async def convert(self, ctx, argument):
            to_slap = random.choice(ctx.guild.members)
            return f'{ctx.author} slapped {to_slap} because *{argument}*'

    @bot.command()
    async def slap(ctx, *, reason: Slapper):
        await ctx.send(reason)

The converter provided can either be constructed or not. Essentially these two are equivalent:

.. code-block:: python3

    @bot.command()
    async def slap(ctx, *, reason: Slapper):
        await ctx.send(reason)

    # is the same as...

    @bot.command()
    async def slap(ctx, *, reason: Slapper()):
        await ctx.send(reason)

Having the possibility of the converter be constructed allows you to set up some state in the converter's ``__init__`` for
fine tuning the converter. An example of this is actually in the library, :class:`~ext.commands.clean_content`.

.. code-block:: python3

    @bot.command()
    async def clean(ctx, *, content: commands.clean_content):
        await ctx.send(content)

    # or for fine-tuning

    @bot.command()
    async def clean(ctx, *, content: commands.clean_content(use_nicknames=False)):
        await ctx.send(content)


If a converter fails to convert an argument to its designated target type, the :exc:`.BadArgument` exception must be
raised.

Inline Advanced Converters
+++++++++++++++++++++++++++++

If we don't want to inherit from :class:`~ext.commands.Converter`, we can still provide a converter that has the
advanced functionalities of an advanced converter and save us from specifying two types.

For example, a common idiom would be to have a class and a converter for that class:

.. code-block:: python3

    class JoinDistance:
        def __init__(self, joined, created):
            self.joined = joined
            self.created = created

        @property
        def delta(self):
            return self.joined - self.created

    class JoinDistanceConverter(commands.MemberConverter):
        async def convert(self, ctx, argument):
            member = await super().convert(ctx, argument)
            return JoinDistance(member.joined_at, member.created_at)

    @bot.command()
    async def delta(ctx, *, member: JoinDistanceConverter):
        is_new = member.delta.days < 100
        if is_new:
            await ctx.send("Hey you're pretty new!")
        else:
            await ctx.send("Hm you're not so new.")

This can get tedious, so an inline advanced converter is possible through a :func:`classmethod` inside the type:

.. code-block:: python3

    class JoinDistance:
        def __init__(self, joined, created):
            self.joined = joined
            self.created = created

        @classmethod
        async def convert(cls, ctx, argument):
            member = await commands.MemberConverter().convert(ctx, argument)
            return cls(member.joined_at, member.created_at)

        @property
        def delta(self):
            return self.joined - self.created

    @bot.command()
    async def delta(ctx, *, member: JoinDistance):
        is_new = member.delta.days < 100
        if is_new:
            await ctx.send("Hey you're pretty new!")
        else:
            await ctx.send("Hm you're not so new.")

Discord Converters
++++++++++++++++++++

Working with :ref:`discord_api_models` is a fairly common thing when defining commands, as a result the library makes
working with them easy.

For example, to receive a :class:`Member` you can just pass it as a converter:

.. code-block:: python3

    @bot.command()
    async def joined(ctx, *, member: discord.Member):
        await ctx.send(f'{member} joined on {member.joined_at}')

When this command is executed, it attempts to convert the string given into a :class:`Member` and then passes it as a
parameter for the function. This works by checking if the string is a mention, an ID, a nickname, a username + discriminator,
or just a regular username. The default set of converters have been written to be as easy to use as possible.

A lot of discord models work out of the gate as a parameter:

- :class:`Object` (since v2.0)
- :class:`Member`
- :class:`User`
- :class:`Message` (since v1.1)
- :class:`PartialMessage` (since v1.7)
- :class:`abc.GuildChannel` (since 2.0)
- :class:`TextChannel`
- :class:`VoiceChannel`
- :class:`StageChannel` (since v1.7)
- :class:`CategoryChannel`
- :class:`ForumChannel` (since v2.0)
- :class:`Invite`
- :class:`Guild` (since v1.7)
- :class:`Role`
- :class:`Game`
- :class:`Colour`
- :class:`Emoji`
- :class:`PartialEmoji`
- :class:`Thread` (since v2.0)
- :class:`GuildSticker` (since v2.0)
- :class:`ScheduledEvent` (since v2.0)

Having any of these set as the converter will intelligently convert the argument to the appropriate target type you
specify.

Under the hood, these are implemented by the :ref:`ext_commands_adv_converters` interface. A table of the equivalent
converter is given below:

+--------------------------+-------------------------------------------------+
|     Discord Class        |                    Converter                    |
+--------------------------+-------------------------------------------------+
| :class:`Object`          | :class:`~ext.commands.ObjectConverter`          |
+--------------------------+-------------------------------------------------+
| :class:`Member`          | :class:`~ext.commands.MemberConverter`          |
+--------------------------+-------------------------------------------------+
| :class:`User`            | :class:`~ext.commands.UserConverter`            |
+--------------------------+-------------------------------------------------+
| :class:`Message`         | :class:`~ext.commands.MessageConverter`         |
+--------------------------+-------------------------------------------------+
| :class:`PartialMessage`  | :class:`~ext.commands.PartialMessageConverter`  |
+--------------------------+-------------------------------------------------+
| :class:`.GuildChannel`   | :class:`~ext.commands.GuildChannelConverter`    |
+--------------------------+-------------------------------------------------+
| :class:`TextChannel`     | :class:`~ext.commands.TextChannelConverter`     |
+--------------------------+-------------------------------------------------+
| :class:`VoiceChannel`    | :class:`~ext.commands.VoiceChannelConverter`    |
+--------------------------+-------------------------------------------------+
| :class:`StageChannel`    | :class:`~ext.commands.StageChannelConverter`    |
+--------------------------+-------------------------------------------------+
| :class:`CategoryChannel` | :class:`~ext.commands.CategoryChannelConverter` |
+--------------------------+-------------------------------------------------+
| :class:`ForumChannel`    | :class:`~ext.commands.ForumChannelConverter`    |
+--------------------------+-------------------------------------------------+
| :class:`Invite`          | :class:`~ext.commands.InviteConverter`          |
+--------------------------+-------------------------------------------------+
| :class:`Guild`           | :class:`~ext.commands.GuildConverter`           |
+--------------------------+-------------------------------------------------+
| :class:`Role`            | :class:`~ext.commands.RoleConverter`            |
+--------------------------+-------------------------------------------------+
| :class:`Game`            | :class:`~ext.commands.GameConverter`            |
+--------------------------+-------------------------------------------------+
| :class:`Colour`          | :class:`~ext.commands.ColourConverter`          |
+--------------------------+-------------------------------------------------+
| :class:`Emoji`           | :class:`~ext.commands.EmojiConverter`           |
+--------------------------+-------------------------------------------------+
| :class:`PartialEmoji`    | :class:`~ext.commands.PartialEmojiConverter`    |
+--------------------------+-------------------------------------------------+
| :class:`Thread`          | :class:`~ext.commands.ThreadConverter`          |
+--------------------------+-------------------------------------------------+
| :class:`GuildSticker`    | :class:`~ext.commands.GuildStickerConverter`    |
+--------------------------+-------------------------------------------------+
| :class:`ScheduledEvent`  | :class:`~ext.commands.ScheduledEventConverter`  |
+--------------------------+-------------------------------------------------+

By providing the converter it allows us to use them as building blocks for another converter:

.. code-block:: python3

    class MemberRoles(commands.MemberConverter):
        async def convert(self, ctx, argument):
            member = await super().convert(ctx, argument)
            return [role.name for role in member.roles[1:]] # Remove everyone role!

    @bot.command()
    async def roles(ctx, *, member: MemberRoles):
        """Tells you a member's roles."""
        await ctx.send('I see the following roles: ' + ', '.join(member))

.. _ext_commands_special_converters:

Special Converters
++++++++++++++++++++

The command extension also has support for certain converters to allow for more advanced and intricate use cases that go
beyond the generic linear parsing. These converters allow you to introduce some more relaxed and dynamic grammar to your
commands in an easy to use manner.

typing.Union
^^^^^^^^^^^^^^

A :data:`typing.Union` is a special type hint that allows for the command to take in any of the specific types instead of
a singular type. For example, given the following:

.. code-block:: python3

    import typing

    @bot.command()
    async def union(ctx, what: typing.Union[discord.TextChannel, discord.Member]):
        await ctx.send(what)


The ``what`` parameter would either take a :class:`discord.TextChannel` converter or a :class:`discord.Member` converter.
The way this works is through a left-to-right order. It first attempts to convert the input to a
:class:`discord.TextChannel`, and if it fails it tries to convert it to a :class:`discord.Member`. If all converters fail,
then a special error is raised, :exc:`~ext.commands.BadUnionArgument`.

Note that any valid converter discussed above can be passed in to the argument list of a :data:`typing.Union`.

typing.Optional
^^^^^^^^^^^^^^^^^

A :data:`typing.Optional` is a special type hint that allows for "back-referencing" behaviour. If the converter fails to
parse into the specified type, the parser will skip the parameter and then either ``None`` or the specified default will be
passed into the parameter instead. The parser will then continue on to the next parameters and converters, if any.

Consider the following example:

.. code-block:: python3

    import typing

    @bot.command()
    async def bottles(ctx, amount: typing.Optional[int] = 99, *, liquid="beer"):
        await ctx.send(f'{amount} bottles of {liquid} on the wall!')


.. image:: /images/commands/optional1.png

In this example, since the argument could not be converted into an ``int``, the default of ``99`` is passed and the parser
resumes handling, which in this case would be to pass it into the ``liquid`` parameter.

.. note::

    This converter only works in regular positional parameters, not variable parameters or keyword-only parameters.

typing.Literal
^^^^^^^^^^^^^^^^

.. versionadded:: 2.0

A :data:`typing.Literal` is a special type hint that requires the passed parameter to be equal to one of the listed values
after being converted to the same type. For example, given the following:

.. code-block:: python3

    from typing import Literal

    @bot.command()
    async def shop(ctx, buy_sell: Literal['buy', 'sell'], amount: Literal[1, 2], *, item: str):
        await ctx.send(f'{buy_sell.capitalize()}ing {amount} {item}(s)!')


The ``buy_sell`` parameter must be either the literal string ``"buy"`` or ``"sell"`` and ``amount`` must convert to the
``int`` ``1`` or ``2``. If ``buy_sell`` or ``amount`` don't match any value, then a special error is raised,
:exc:`~.ext.commands.BadLiteralArgument`. Any literal values can be mixed and matched within the same :data:`typing.Literal` converter.

Note that ``typing.Literal[True]`` and ``typing.Literal[False]`` still follow the :class:`bool` converter rules.

typing.Annotated
^^^^^^^^^^^^^^^^^

.. versionadded:: 2.0

A :data:`typing.Annotated` is a special type introduced in Python 3.9 that allows the type checker to see one type, but allows the library to see another type. This is useful for appeasing the type checker for complicated converters. The second parameter of ``Annotated`` must be the converter that the library should use.

For example, given the following:

.. code-block:: python3

    from typing import Annotated

    @bot.command()
    async def fun(ctx, arg: Annotated[str, lambda s: s.upper()]):
        await ctx.send(arg)

The type checker will see ``arg`` as a regular :class:`str` but the library will know you wanted to change the input into all upper-case.

.. note::

    For Python versions below 3.9, it is recommended to install the ``typing_extensions`` library and import ``Annotated`` from there.


Greedy
^^^^^^^^

The :class:`~ext.commands.Greedy` converter is a generalisation of the :data:`typing.Optional` converter, except applied
to a list of arguments. In simple terms, this means that it tries to convert as much as it can until it can't convert
any further.

Consider the following example:

.. code-block:: python3

    @bot.command()
    async def slap(ctx, members: commands.Greedy[discord.Member], *, reason='no reason'):
        slapped = ", ".join(x.name for x in members)
        await ctx.send(f'{slapped} just got slapped for {reason}')

When invoked, it allows for any number of members to be passed in:

.. image:: /images/commands/greedy1.png

The type passed when using this converter depends on the parameter type that it is being attached to:

- Positional parameter types will receive either the default parameter or a :class:`list` of the converted values.
- Variable parameter types will be a :class:`tuple` as usual.
- Keyword-only parameter types will be the same as if :class:`~ext.commands.Greedy` was not passed at all.

:class:`~ext.commands.Greedy` parameters can also be made optional by specifying an optional value.

When mixed with the :data:`typing.Optional` converter you can provide simple and expressive command invocation syntaxes:

.. code-block:: python3

    import typing

    @bot.command()
    async def ban(ctx, members: commands.Greedy[discord.Member],
                       delete_days: typing.Optional[int] = 0, *,
                       reason: str):
        """Mass bans members with an optional delete_days parameter"""
        delete_seconds = delete_days * 86400 # one day
        for member in members:
            await member.ban(delete_message_seconds=delete_seconds, reason=reason)


This command can be invoked any of the following ways:

.. code-block:: none

    $ban @Member @Member2 spam bot
    $ban @Member @Member2 7 spam bot
    $ban @Member spam

.. warning::

    The usage of :class:`~ext.commands.Greedy` and :data:`typing.Optional` are powerful and useful, however as a
    price, they open you up to some parsing ambiguities that might surprise some people.

    For example, a signature expecting a :data:`typing.Optional` of a :class:`discord.Member` followed by a
    :class:`int` could catch a member named after a number due to the different ways a
    :class:`~ext.commands.MemberConverter` decides to fetch members. You should take care to not introduce
    unintended parsing ambiguities in your code. One technique would be to clamp down the expected syntaxes
    allowed through custom converters or reordering the parameters to minimise clashes.

    To help aid with some parsing ambiguities, :class:`str`, ``None``, :data:`typing.Optional` and
    :class:`~ext.commands.Greedy` are forbidden as parameters for the :class:`~ext.commands.Greedy` converter.


discord.Attachment
^^^^^^^^^^^^^^^^^^^

.. versionadded:: 2.0

The :class:`discord.Attachment` converter is a special converter that retrieves an attachment from the uploaded attachments on a message. This converter *does not* look at the message content at all and just the uploaded attachments.

Consider the following example:

.. code-block:: python3

    import discord

    @bot.command()
    async def upload(ctx, attachment: discord.Attachment):
        await ctx.send(f'You have uploaded <{attachment.url}>')


When this command is invoked, the user must directly upload a file for the command body to be executed. When combined with the :data:`typing.Optional` converter, the user does not have to provide an attachment.

.. code-block:: python3

    import typing
    import discord

    @bot.command()
    async def upload(ctx, attachment: typing.Optional[discord.Attachment]):
        if attachment is None:
            await ctx.send('You did not upload anything!')
        else:
            await ctx.send(f'You have uploaded <{attachment.url}>')


This also works with multiple attachments:

.. code-block:: python3

    import typing
    import discord

    @bot.command()
    async def upload_many(
        ctx,
        first: discord.Attachment,
        second: typing.Optional[discord.Attachment],
    ):
        if second is None:
            files = [first.url]
        else:
            files = [first.url, second.url]

        await ctx.send(f'You uploaded: {" ".join(files)}')


In this example the user must provide at least one file but the second one is optional.

As a special case, using :class:`~ext.commands.Greedy` will return the remaining attachments in the message, if any.

.. code-block:: python3

    import discord
    from discord.ext import commands

    @bot.command()
    async def upload_many(
        ctx,
        first: discord.Attachment,
        remaining: commands.Greedy[discord.Attachment],
    ):
        files = [first.url]
        files.extend(a.url for a in remaining)
        await ctx.send(f'You uploaded: {" ".join(files)}')


Note that using a :class:`discord.Attachment` converter after a :class:`~ext.commands.Greedy` of :class:`discord.Attachment` will always fail since the greedy had already consumed the remaining attachments.

If an attachment is expected but not given, then :exc:`~ext.commands.MissingRequiredAttachment` is raised to the error handlers.

.. _ext_commands_flag_converter:

FlagConverter
++++++++++++++

.. versionadded:: 2.0

A :class:`~ext.commands.FlagConverter` allows the user to specify user-friendly "flags" using :pep:`526` type annotations
or a syntax more reminiscent of the :mod:`py:dataclasses` module.

For example, the following code:

.. code-block:: python3

    from discord.ext import commands
    import discord

    class BanFlags(commands.FlagConverter):
        member: discord.Member
        reason: str
        days: int = 1

    @commands.command()
    async def ban(ctx, *, flags: BanFlags):
        plural = f'{flags.days} days' if flags.days != 1 else f'{flags.days} day'
        await ctx.send(f'Banned {flags.member} for {flags.reason!r} (deleted {plural} worth of messages)')

Allows the user to invoke the command using a simple flag-like syntax:

.. image:: /images/commands/flags1.png

Flags use a syntax that allows the user to not require quotes when passing in values to the flag. The goal of the
flag syntax is to be as user-friendly as possible. This makes flags a good choice for complicated commands that can have
multiple knobs to turn or simulating keyword-only parameters in your external command interface. **It is recommended to use
keyword-only parameters with the flag converter**. This ensures proper parsing and behaviour with quoting.

Internally, the :class:`~ext.commands.FlagConverter` class examines the class to find flags. A flag can either be a
class variable with a type annotation or a class variable that's been assigned the result of the :func:`~ext.commands.flag`
function. These flags are then used to define the interface that your users will use. The annotations correspond to
the converters that the flag arguments must adhere to.

For most use cases, no extra work is required to define flags. However, if customisation is needed to control the flag name
or the default value then the :func:`~ext.commands.flag` function can come in handy:

.. code-block:: python3

    from typing import List

    class BanFlags(commands.FlagConverter):
        members: List[discord.Member] = commands.flag(name='member', default=lambda ctx: [])

This tells the parser that the ``members`` attribute is mapped to a flag named ``member`` and that
the default value is an empty list. For greater customisability, the default can either be a value or a callable
that takes the :class:`~ext.commands.Context` as a sole parameter. This callable can either be a function or a coroutine.

A positional flag can be defined by setting the :attr:`~ext.commands.Flag.positional` attribute to ``True``. This
tells the parser that the content provided before the parsing occurs is part of the flag. This is useful for commands that
require a parameter to be used first and the flags are optional, such as the following:

.. code-block:: python3

    class BanFlags(commands.FlagConverter):
        members: List[discord.Member] = commands.flag(name='member', positional=True, default=lambda ctx: [])
        reason: Optional[str] = None

.. note::
    Only one positional flag is allowed in a flag converter.

In order to customise the flag syntax we also have a few options that can be passed to the class parameter list:

.. code-block:: python3

    # --hello world syntax
    class PosixLikeFlags(commands.FlagConverter, delimiter=' ', prefix='--'):
        hello: str


    # /make food
    class WindowsLikeFlags(commands.FlagConverter, prefix='/', delimiter=''):
        make: str

    # TOPIC: not allowed nsfw: yes Slowmode: 100
    class Settings(commands.FlagConverter, case_insensitive=True):
        topic: Optional[str]
        nsfw: Optional[bool]
        slowmode: Optional[int]
    
    # Hello there --bold True
    class Greeting(commands.FlagConverter):
        text: str = commands.flag(positional=True)
        bold: bool = False

.. note::

    Despite the similarities in these examples to command like arguments, the syntax and parser is not
    a command line parser. The syntax is mainly inspired by Discord's search bar input and as a result
    all flags need a corresponding value unless part of a positional flag.

Flag converters will only raise :exc:`~ext.commands.FlagError` derived exceptions. If an error is raised while
converting a flag, :exc:`~ext.commands.BadFlagArgument` is raised instead and the original exception
can be accessed with the :attr:`~ext.commands.BadFlagArgument.original` attribute.

The flag converter is similar to regular commands and allows you to use most types of converters
(with the exception of :class:`~ext.commands.Greedy`) as the type annotation. Some extra support is added for specific
annotations as described below.

typing.List
^^^^^^^^^^^^^

If a list is given as a flag annotation it tells the parser that the argument can be passed multiple times.

For example, augmenting the example above:

.. code-block:: python3

    from discord.ext import commands
    from typing import List
    import discord

    class BanFlags(commands.FlagConverter):
        members: List[discord.Member] = commands.flag(name='member')
        reason: str
        days: int = 1

    @commands.command()
    async def ban(ctx, *, flags: BanFlags):
        for member in flags.members:
            await member.ban(reason=flags.reason, delete_message_days=flags.days)

        members = ', '.join(str(member) for member in flags.members)
        plural = f'{flags.days} days' if flags.days != 1 else f'{flags.days} day'
        await ctx.send(f'Banned {members} for {flags.reason!r} (deleted {plural} worth of messages)')

This is called by repeatedly specifying the flag:

.. image:: /images/commands/flags2.png

typing.Tuple
^^^^^^^^^^^^^

Since the above syntax can be a bit repetitive when specifying a flag many times, the :class:`py:tuple` type annotation
allows for "greedy-like" semantics using a variadic tuple:

.. code-block:: python3

    from discord.ext import commands
    from typing import Tuple
    import discord

    class BanFlags(commands.FlagConverter):
        members: Tuple[discord.Member, ...]
        reason: str
        days: int = 1

This allows the previous ``ban`` command to be called like this:

.. image:: /images/commands/flags3.png

The :class:`py:tuple` annotation also allows for parsing of pairs. For example, given the following code:

.. code-block:: python3

    # point: 10 11 point: 12 13
    class Coordinates(commands.FlagConverter):
        point: Tuple[int, int]


.. warning::

    Due to potential parsing ambiguities, the parser expects tuple arguments to be quoted
    if they require spaces. So if one of the inner types is :class:`str` and the argument requires spaces
    then quotes should be used to disambiguate it from the other element of the tuple.

typing.Dict
^^^^^^^^^^^^^

A :class:`dict` annotation is functionally equivalent to ``List[Tuple[K, V]]`` except with the return type
given as a :class:`dict` rather than a :class:`list`.


Hybrid Command Interaction
^^^^^^^^^^^^^^^^^^^^^^^^^^^

When used as a hybrid command, the parameters are flattened into different parameters for the application command. For example, the following converter:

.. code-block:: python3

    class BanFlags(commands.FlagConverter):
        member: discord.Member
        reason: str
        days: int = 1


    @commands.hybrid_command()
    async def ban(ctx, *, flags: BanFlags):
        ...

Would be equivalent to an application command defined as this:

.. code-block:: python3

    @commands.hybrid_command()
    async def ban(ctx, member: discord.Member, reason: str, days: int = 1):
        ...

This means that decorators that refer to a parameter by name will use the flag name instead:

.. code-block:: python3

    class BanFlags(commands.FlagConverter):
        member: discord.Member
        reason: str
        days: int = 1


    @commands.hybrid_command()
    @app_commands.describe(
        member='The member to ban',
        reason='The reason for the ban',
        days='The number of days worth of messages to delete',
    )
    async def ban(ctx, *, flags: BanFlags):
        ...

For ease of use, the :func:`~ext.commands.flag` function accepts a ``description`` keyword argument to allow you to pass descriptions inline:

.. code-block:: python3

    class BanFlags(commands.FlagConverter):
        member: discord.Member = commands.flag(description='The member to ban')
        reason: str = commands.flag(description='The reason for the ban')
        days: int = commands.flag(default=1, description='The number of days worth of messages to delete')


    @commands.hybrid_command()
    async def ban(ctx, *, flags: BanFlags):
        ...


Likewise, use of the ``name`` keyword argument allows you to pass renames for the parameter, similar to the :func:`~discord.app_commands.rename` decorator.

Note that in hybrid command form, a few annotations are unsupported due to Discord limitations:

- ``typing.Tuple``
- ``typing.List``
- ``typing.Dict``

.. note::

    Only one flag converter is supported per hybrid command. Due to the flag converter's way of working, it is unlikely for a user to have two of them in one signature.

.. _ext_commands_parameter:

Parameter Metadata
-------------------

:func:`~ext.commands.parameter` assigns custom metadata to a :class:`~ext.commands.Command`'s parameter.

This is useful for:

- Custom converters as annotating a parameter with a custom converter works at runtime, type checkers don't like it
  because they can't understand what's going on.

  .. code-block:: python3

      class SomeType:
          foo: int

      class MyVeryCoolConverter(commands.Converter[SomeType]):
          ...  # implementation left as an exercise for the reader

      @bot.command()
      async def bar(ctx, cool_value: MyVeryCoolConverter):
          cool_value.foo  # type checker warns MyVeryCoolConverter has no value foo (uh-oh)

  However, fear not we can use :func:`~ext.commands.parameter` to tell type checkers what's going on.

  .. code-block:: python3

      @bot.command()
      async def bar(ctx, cool_value: SomeType = commands.parameter(converter=MyVeryCoolConverter)):
          cool_value.foo  # no error (hurray)

- Late binding behaviour

  .. code-block:: python3

      @bot.command()
      async def wave(ctx, to: discord.User = commands.parameter(default=lambda ctx: ctx.author)):
          await ctx.send(f'Hello {to.mention} :wave:')

  Because this is such a common use-case, the library provides :obj:`~.ext.commands.Author`, :obj:`~.ext.commands.CurrentChannel` and
  :obj:`~.ext.commands.CurrentGuild`, armed with this we can simplify ``wave`` to:

  .. code-block:: python3

      @bot.command()
      async def wave(ctx, to: discord.User = commands.Author):
          await ctx.send(f'Hello {to.mention} :wave:')

  :obj:`~.ext.commands.Author` and co also have other benefits like having the displayed default being filled.


.. _ext_commands_error_handler:

Error Handling
----------------

When our commands fail to parse we will, by default, receive a noisy error in ``stderr`` of our console that tells us
that an error has happened and has been silently ignored.

In order to handle our errors, we must use something called an error handler. There is a global error handler, called
:func:`.on_command_error` which works like any other event in the :ref:`discord-api-events`. This global error handler is
called for every error reached.

Most of the time however, we want to handle an error local to the command itself. Luckily, commands come with local error
handlers that allow us to do just that. First we decorate an error handler function with :meth:`~discord.ext.commands.Command.error`:

.. code-block:: python3

    @bot.command()
    async def info(ctx, *, member: discord.Member):
        """Tells you some info about the member."""
        msg = f'{member} joined on {member.joined_at} and has {len(member.roles)} roles.'
        await ctx.send(msg)

    @info.error
    async def info_error(ctx, error):
        if isinstance(error, commands.BadArgument):
            await ctx.send('I could not find that member...')

The first parameter of the error handler is the :class:`.Context` while the second one is an exception that is derived from
:exc:`~ext.commands.CommandError`. A list of errors is found in the :ref:`ext_commands_api_errors` page of the documentation.

Checks
-------

There are cases when we don't want a user to use our commands. They don't have permissions to do so or maybe we blocked
them from using our bot earlier. The commands extension comes with full support for these things in a concept called a
:ref:`ext_commands_api_checks`.

A check is a basic predicate that can take in a :class:`.Context` as its sole parameter. Within it, you have the following
options:

- Return ``True`` to signal that the person can run the command.
- Return ``False`` to signal that the person cannot run the command.
- Raise a :exc:`~ext.commands.CommandError` derived exception to signal the person cannot run the command.

    - This allows you to have custom error messages for you to handle in the
      :ref:`error handlers <ext_commands_error_handler>`.

To register a check for a command, we would have two ways of doing so. The first is using the :meth:`~ext.commands.check`
decorator. For example:

.. code-block:: python3

    async def is_owner(ctx):
        return ctx.author.id == 316026178463072268

    @bot.command(name='eval')
    @commands.check(is_owner)
    async def _eval(ctx, *, code):
        """A bad example of an eval command"""
        await ctx.send(eval(code))

This would only evaluate the command if the function ``is_owner`` returns ``True``. Sometimes we re-use a check often and
want to split it into its own decorator. To do that we can just add another level of depth:

.. code-block:: python3

    def is_owner():
        async def predicate(ctx):
            return ctx.author.id == 316026178463072268
        return commands.check(predicate)

    @bot.command(name='eval')
    @is_owner()
    async def _eval(ctx, *, code):
        """A bad example of an eval command"""
        await ctx.send(eval(code))


Since an owner check is so common, the library provides it for you (:func:`~ext.commands.is_owner`):

.. code-block:: python3

    @bot.command(name='eval')
    @commands.is_owner()
    async def _eval(ctx, *, code):
        """A bad example of an eval command"""
        await ctx.send(eval(code))

When multiple checks are specified, **all** of them must be ``True``:

.. code-block:: python3

    def is_in_guild(guild_id):
        async def predicate(ctx):
            return ctx.guild and ctx.guild.id == guild_id
        return commands.check(predicate)

    @bot.command()
    @commands.is_owner()
    @is_in_guild(41771983423143937)
    async def secretguilddata(ctx):
        """super secret stuff"""
        await ctx.send('secret stuff')

If any of those checks fail in the example above, then the command will not be run.

When an error happens, the error is propagated to the :ref:`error handlers <ext_commands_error_handler>`. If you do not
raise a custom :exc:`~ext.commands.CommandError` derived exception, then it will get wrapped up into a
:exc:`~ext.commands.CheckFailure` exception as so:

.. code-block:: python3

    @bot.command()
    @commands.is_owner()
    @is_in_guild(41771983423143937)
    async def secretguilddata(ctx):
        """super secret stuff"""
        await ctx.send('secret stuff')

    @secretguilddata.error
    async def secretguilddata_error(ctx, error):
        if isinstance(error, commands.CheckFailure):
            await ctx.send('nothing to see here comrade.')

If you want a more robust error system, you can derive from the exception and raise it instead of returning ``False``:

.. code-block:: python3

    class NoPrivateMessages(commands.CheckFailure):
        pass

    def guild_only():
        async def predicate(ctx):
            if ctx.guild is None:
                raise NoPrivateMessages('Hey no DMs!')
            return True
        return commands.check(predicate)

    @bot.command()
    @guild_only()
    async def test(ctx):
        await ctx.send('Hey this is not a DM! Nice.')

    @test.error
    async def test_error(ctx, error):
        if isinstance(error, NoPrivateMessages):
            await ctx.send(error)

.. note::

    Since having a ``guild_only`` decorator is pretty common, it comes built-in via :func:`~ext.commands.guild_only`.

Global Checks
++++++++++++++

Sometimes we want to apply a check to **every** command, not just certain commands. The library supports this as well
using the global check concept.

Global checks work similarly to regular checks except they are registered with the :meth:`.Bot.check` decorator.

For example, to block all DMs we could do the following:

.. code-block:: python3

    @bot.check
    async def globally_block_dms(ctx):
        return ctx.guild is not None

.. warning::

    Be careful on how you write your global checks, as it could also lock you out of your own bot.

.. need a note on global check once here I think


Hybrid Commands
---------------

.. versionadded:: 2.0

:class:`.commands.HybridCommand` is a command that can be invoked as both a text and a slash command.
This allows you to define a command as both slash and text command without writing separate code for
both counterparts.


In order to define a hybrid command, The command callback should be decorated with
:meth:`.Bot.hybrid_command` decorator.

.. code-block:: python3

    @bot.hybrid_command()
    async def test(ctx):
        await ctx.send("This is a hybrid command!")

The above command can be invoked as both text and slash command. Note that you have to manually
sync your :class:`~app_commands.CommandTree` by calling :class:`~app_commands.CommandTree.sync` in order
for slash commands to appear.

.. image:: /images/commands/hybrid1.png
.. image:: /images/commands/hybrid2.png

You can create hybrid command groups and sub-commands using the :meth:`.Bot.hybrid_group`
decorator.

.. code-block:: python3

    @bot.hybrid_group(fallback="get")
    async def tag(ctx, name):
        await ctx.send(f"Showing tag: {name}")

    @tag.command()
    async def create(ctx, name):
        await ctx.send(f"Created tag: {name}")

Due to a Discord limitation, slash command groups cannot be invoked directly so the ``fallback``
parameter allows you to create a sub-command that will be bound to callback of parent group.

.. image:: /images/commands/hybrid3.png
.. image:: /images/commands/hybrid4.png

Due to certain limitations on slash commands, some features of text commands are not supported
on hybrid commands. You can define a hybrid command as long as it meets the same subset that is
supported for slash commands.

Following are currently **not supported** by hybrid commands:

- Variable number of arguments. e.g. ``*arg: int``
- Group commands with a depth greater than 1.
- Most :class:`typing.Union` types.
    - Unions of channel types are allowed
    - Unions of user types are allowed
    - Unions of user types with roles are allowed

Apart from that, all other features such as converters, checks, autocomplete, flags etc.
are supported on hybrid commands. Note that due to a design constraint, decorators related to application commands
such as :func:`discord.app_commands.autocomplete` should be placed below the :func:`~ext.commands.hybrid_command` decorator.

For convenience and ease in writing code, The :class:`~ext.commands.Context` class implements
some behavioural changes for various methods and attributes:

- :attr:`.Context.interaction` can be used to retrieve the slash command interaction.
- Since interaction can only be responded to once, The :meth:`.Context.send` automatically
  determines whether to send an interaction response or a followup response.
- :meth:`.Context.defer` defers the interaction response for slash commands but shows typing
  indicator for text commands.