@ -0,0 +1,40 @@ |
|||||
|
body { |
||||
|
font-family: Georgia, 'Hiragino Mincho Pro', serif; |
||||
|
font-size: 16px; |
||||
|
} |
||||
|
|
||||
|
pre, code { |
||||
|
font-family: 'Consolas', 'Menlo', 'Deja Vu Sans Mono', 'Bitstream Vera Sans Mono', monospace; |
||||
|
font-size: 0.9em; |
||||
|
} |
||||
|
|
||||
|
code.descname, code.descclassname { |
||||
|
font-size: 0.95em; |
||||
|
} |
||||
|
|
||||
|
code.descname { |
||||
|
background-color: transparent; |
||||
|
font-weight: bold; |
||||
|
} |
||||
|
|
||||
|
pre, * pre { |
||||
|
padding: 7px 0 7px 30px !important; |
||||
|
margin: 15px 0 !important; |
||||
|
line-height: 1.3; |
||||
|
} |
||||
|
|
||||
|
div.warning { |
||||
|
background-color: #ffe6cc; |
||||
|
border: 1px solid #ffd5aa; |
||||
|
} |
||||
|
|
||||
|
/* don't link-ify the FAQ page */ |
||||
|
a.toc-backref { |
||||
|
text-decoration: none; |
||||
|
color: #3E4349; |
||||
|
} |
||||
|
|
||||
|
code.xref { |
||||
|
background-color: #ecf0f3; |
||||
|
border-bottom: 1px dotted #222; |
||||
|
} |
@ -0,0 +1,92 @@ |
|||||
|
.. _discord-intro: |
||||
|
|
||||
|
Creating a Bot Account |
||||
|
======================== |
||||
|
|
||||
|
In order to work with the library and the Discord API in general, we must first create a Discord Bot account. |
||||
|
|
||||
|
Creating a Bot account is a pretty straightforward process. |
||||
|
|
||||
|
1. Make sure you're logged on to the `Discord website <https://discordapp.com>`_. |
||||
|
2. Navigate to the `application page <https://discordapp.com/developers/applications/me>`_ |
||||
|
3. Click on the "New App" button. |
||||
|
|
||||
|
.. image:: /images/discord_new_app_button.png |
||||
|
:alt: The new app button. |
||||
|
|
||||
|
4. Give the application a name and a description if wanted and click "Create App". |
||||
|
|
||||
|
- You can also put an avatar you want your bot to use, don't worry you can change this later. |
||||
|
- **Leave the Redirect URI(s) blank** unless are creating a service. |
||||
|
|
||||
|
.. image:: /images/discord_new_app_form.png |
||||
|
:alt: The new application form filled in. |
||||
|
5. Create a Bot User by clicking on the accompanying button and confirming it. |
||||
|
|
||||
|
.. image:: /images/discord_create_bot_user_button.png |
||||
|
:alt: The Create a Bot User button. |
||||
|
6. Make sure that **Public Bot** is ticked if you want others to invite your bot. |
||||
|
|
||||
|
- You should also make sure that **Require OAuth2 Code Grant** is unchecked unless you |
||||
|
are developing a service that needs it. If you're unsure, then **leave it unchecked**. |
||||
|
|
||||
|
.. figure:: /images/discord_finished_bot_user.png |
||||
|
|
||||
|
How the Bot User options should look like for most people. |
||||
|
|
||||
|
7. Click to reveal the token. |
||||
|
|
||||
|
- **This is not the Client Secret** |
||||
|
|
||||
|
.. figure:: /images/discord_reveal_token.png |
||||
|
|
||||
|
How the token reveal button looks like. |
||||
|
|
||||
|
And that's it. You now have a bot account and you can login with that token. |
||||
|
|
||||
|
.. _discord_invite_bot: |
||||
|
|
||||
|
Inviting Your Bot |
||||
|
------------------- |
||||
|
|
||||
|
So you've made a Bot User but it's not actually in any server. |
||||
|
|
||||
|
If you want to invite your bot you must create an invite URL for your bot. |
||||
|
|
||||
|
First, you must fetch the Client ID of the Bot. You can find this in the Bot's application page. |
||||
|
|
||||
|
.. image:: /images/discord_client_id.png |
||||
|
:alt: The Bot's Client ID. |
||||
|
|
||||
|
Copy paste that into the pre-formatted URL: |
||||
|
|
||||
|
.. code-block:: none |
||||
|
|
||||
|
https://discordapp.com/oauth2/authorize?client_id=YOUR_CLIENT_ID&scope=bot&permissions=0 |
||||
|
|
||||
|
Replace ``YOUR_CLIENT_ID`` with the Client ID we got in the previous step. For example, |
||||
|
in the image above our client ID is 312718641634213889 so the resulting URL would be |
||||
|
https://discordapp.com/oauth2/authorize?client_id=312718641634213889&scope=bot&permissions=0 |
||||
|
(note that this bot has been deleted). |
||||
|
|
||||
|
Now you can click the link and invite your bot to any server you have "Manage Server" permissions on. |
||||
|
|
||||
|
Adding Permissions |
||||
|
~~~~~~~~~~~~~~~~~~~~ |
||||
|
|
||||
|
In the above URL, you might have noticed an interesting bit, the ``permissions=0`` fragment. |
||||
|
|
||||
|
Bot accounts can request specific permissions to be granted upon joining. When the bot joins |
||||
|
the guild, they will be granted a managed role that contains the permissions you requested. |
||||
|
If the permissions is 0, then no special role is created. |
||||
|
|
||||
|
This ``permissions`` value is calculated based on bit-wise arithmetic. Thankfully, people have |
||||
|
created a calculate that makes it easy to calculate the permissions necessary visually. |
||||
|
|
||||
|
- https://discordapi.com/permissions.html |
||||
|
- https://finitereality.github.io/permissions/ |
||||
|
|
||||
|
Feel free to use whichever is easier for you to grasp. |
||||
|
|
||||
|
If you want to generate this URL dynamically at run-time inside your bot and using the |
||||
|
:class:`discord.Permissions` interface, you can use :func:`discord.utils.oauth_url`. |
@ -0,0 +1,197 @@ |
|||||
|
.. currentmodule:: discord |
||||
|
|
||||
|
API Reference |
||||
|
=============== |
||||
|
|
||||
|
The following section outlines the API of discord.py's command extension module. |
||||
|
|
||||
|
Bot |
||||
|
---- |
||||
|
|
||||
|
.. autoclass:: discord.ext.commands.Bot |
||||
|
:members: |
||||
|
:inherited-members: |
||||
|
|
||||
|
.. autoclass:: discord.ext.commands.AutoShardedBot |
||||
|
:members: |
||||
|
|
||||
|
Event Reference |
||||
|
----------------- |
||||
|
|
||||
|
These events function similar to :ref:`the regular events <discord-api-events>`, except they |
||||
|
are custom to the command extension module. |
||||
|
|
||||
|
.. function:: on_command_error(ctx, error) |
||||
|
|
||||
|
An error handler that is called when an error is raised |
||||
|
inside a command either through user input error, check |
||||
|
failure, or an error in your own code. |
||||
|
|
||||
|
A default one is provided (:meth:`.Bot.on_command_error`). |
||||
|
|
||||
|
:param ctx: The invocation context. |
||||
|
:type ctx: :class:`Context` |
||||
|
:param error: The error that was raised. |
||||
|
:type error: :class:`CommandError` derived |
||||
|
|
||||
|
.. function:: on_command(ctx) |
||||
|
|
||||
|
An event that is called when a command is found and is about to be invoked. |
||||
|
|
||||
|
This event is called regardless of whether the command itself succeeds via |
||||
|
error or completes. |
||||
|
|
||||
|
:param ctx: The invocation context. |
||||
|
:type ctx: :class:`Context` |
||||
|
|
||||
|
.. function:: on_command_completion(ctx) |
||||
|
|
||||
|
An event that is called when a command has completed its invocation. |
||||
|
|
||||
|
This event is called only if the command succeeded, i.e. all checks have |
||||
|
passed and the user input it correctly. |
||||
|
|
||||
|
:param ctx: The invocation context. |
||||
|
:type ctx: :class:`Context` |
||||
|
|
||||
|
|
||||
|
Command |
||||
|
-------- |
||||
|
|
||||
|
.. autofunction:: discord.ext.commands.command |
||||
|
|
||||
|
.. autofunction:: discord.ext.commands.group |
||||
|
|
||||
|
.. autoclass:: discord.ext.commands.Command |
||||
|
:members: |
||||
|
|
||||
|
.. autoclass:: discord.ext.commands.Group |
||||
|
:members: |
||||
|
:inherited-members: |
||||
|
|
||||
|
.. autoclass:: discord.ext.commands.GroupMixin |
||||
|
:members: |
||||
|
|
||||
|
|
||||
|
Formatters |
||||
|
----------- |
||||
|
|
||||
|
.. autoclass:: discord.ext.commands.Paginator |
||||
|
:members: |
||||
|
|
||||
|
.. autoclass:: discord.ext.commands.HelpFormatter |
||||
|
:members: |
||||
|
|
||||
|
Checks |
||||
|
------- |
||||
|
|
||||
|
.. autofunction:: discord.ext.commands.check |
||||
|
|
||||
|
.. autofunction:: discord.ext.commands.has_role |
||||
|
|
||||
|
.. autofunction:: discord.ext.commands.has_permissions |
||||
|
|
||||
|
.. autofunction:: discord.ext.commands.has_any_role |
||||
|
|
||||
|
.. autofunction:: discord.ext.commands.bot_has_role |
||||
|
|
||||
|
.. autofunction:: discord.ext.commands.bot_has_permissions |
||||
|
|
||||
|
.. autofunction:: discord.ext.commands.bot_has_any_role |
||||
|
|
||||
|
.. autofunction:: discord.ext.commands.cooldown |
||||
|
|
||||
|
.. autofunction:: discord.ext.commands.guild_only |
||||
|
|
||||
|
.. autofunction:: discord.ext.commands.is_owner |
||||
|
|
||||
|
.. autofunction:: discord.ext.commands.is_nsfw |
||||
|
|
||||
|
Context |
||||
|
-------- |
||||
|
|
||||
|
.. autoclass:: discord.ext.commands.Context |
||||
|
:members: |
||||
|
:exclude-members: history typing |
||||
|
|
||||
|
.. autocomethod:: discord.ext.commands.Context.history |
||||
|
:async-for: |
||||
|
|
||||
|
.. autocomethod:: discord.ext.commands.Context.typing |
||||
|
:async-with: |
||||
|
|
||||
|
Converters |
||||
|
------------ |
||||
|
|
||||
|
.. autoclass:: discord.ext.commands.Converter |
||||
|
:members: |
||||
|
|
||||
|
.. autoclass:: discord.ext.commands.MemberConverter |
||||
|
:members: |
||||
|
|
||||
|
.. autoclass:: discord.ext.commands.UserConverter |
||||
|
:members: |
||||
|
|
||||
|
.. autoclass:: discord.ext.commands.TextChannelConverter |
||||
|
:members: |
||||
|
|
||||
|
.. autoclass:: discord.ext.commands.InviteConverter |
||||
|
:members: |
||||
|
|
||||
|
.. autoclass:: discord.ext.commands.RoleConverter |
||||
|
:members: |
||||
|
|
||||
|
.. autoclass:: discord.ext.commands.GameConverter |
||||
|
:members: |
||||
|
|
||||
|
.. autoclass:: discord.ext.commands.ColourConverter |
||||
|
:members: |
||||
|
|
||||
|
.. autoclass:: discord.ext.commands.VoiceChannelConverter |
||||
|
:members: |
||||
|
|
||||
|
.. autoclass:: discord.ext.commands.EmojiConverter |
||||
|
:members: |
||||
|
|
||||
|
.. autoclass:: discord.ext.commands.clean_content |
||||
|
:members: |
||||
|
|
||||
|
Errors |
||||
|
------- |
||||
|
|
||||
|
.. autoexception:: discord.ext.commands.CommandError |
||||
|
:members: |
||||
|
|
||||
|
.. autoexception:: discord.ext.commands.MissingRequiredArgument |
||||
|
:members: |
||||
|
|
||||
|
.. autoexception:: discord.ext.commands.BadArgument |
||||
|
:members: |
||||
|
|
||||
|
.. autoexception:: discord.ext.commands.NoPrivateMessage |
||||
|
:members: |
||||
|
|
||||
|
.. autoexception:: discord.ext.commands.CheckFailure |
||||
|
:members: |
||||
|
|
||||
|
.. autoexception:: discord.ext.commands.CommandNotFound |
||||
|
:members: |
||||
|
|
||||
|
.. autoexception:: discord.ext.commands.DisabledCommand |
||||
|
:members: |
||||
|
|
||||
|
.. autoexception:: discord.ext.commands.CommandInvokeError |
||||
|
:members: |
||||
|
|
||||
|
.. autoexception:: discord.ext.commands.TooManyArguments |
||||
|
:members: |
||||
|
|
||||
|
.. autoexception:: discord.ext.commands.UserInputError |
||||
|
:members: |
||||
|
|
||||
|
.. autoexception:: discord.ext.commands.CommandOnCooldown |
||||
|
:members: |
||||
|
|
||||
|
.. autoexception:: discord.ext.commands.NotOwner |
||||
|
:members: |
||||
|
|
@ -0,0 +1,13 @@ |
|||||
|
``discord.ext.commands`` -- Bot commands framework |
||||
|
==================================================== |
||||
|
|
||||
|
``discord.py`` offers a lower level aspect on interacting with Discord. Often times, the library is used for the creation of |
||||
|
bots. However this task can be daunting and confusing to get correctly the first time. Many times there comes a repetition in |
||||
|
creating a bot command framework that is extensible, flexible, and powerful. For this reason, ``discord.py`` comes with an |
||||
|
extension library that handles this for you. |
||||
|
|
||||
|
|
||||
|
.. toctree:: |
||||
|
:maxdepth: 1 |
||||
|
|
||||
|
api |
After Width: | Height: | Size: 5.5 KiB |
After Width: | Height: | Size: 32 KiB |
After Width: | Height: | Size: 25 KiB |
After Width: | Height: | Size: 3.8 KiB |
After Width: | Height: | Size: 34 KiB |
After Width: | Height: | Size: 5.6 KiB |
After Width: | Height: | Size: 28 KiB |
@ -0,0 +1,112 @@ |
|||||
|
.. currentmodule:: discord |
||||
|
|
||||
|
.. _intro: |
||||
|
|
||||
|
Introduction |
||||
|
============== |
||||
|
|
||||
|
This is the documentation for discord.py, a library for Python to aid |
||||
|
in creating applications that utilise the Discord API. |
||||
|
|
||||
|
Prerequisites |
||||
|
--------------- |
||||
|
|
||||
|
discord.py works with Python 3.4.2 or higher. Support for earlier versions of Python |
||||
|
is not provided. Python 2.7 or lower is not supported. Python 3.3 is not supported |
||||
|
due to one of the dependencies (``aiohttp``) not supporting Python 3.3. |
||||
|
|
||||
|
|
||||
|
.. _installing: |
||||
|
|
||||
|
Installing |
||||
|
----------- |
||||
|
|
||||
|
You can get the library directly from PyPI: :: |
||||
|
|
||||
|
python3 -m pip install -U discord.py |
||||
|
|
||||
|
If you are using Windows, then the following should be used instead: :: |
||||
|
|
||||
|
py -3 -m pip install -U discord.py |
||||
|
|
||||
|
|
||||
|
To get voice support, you should use ``discord.py[voice]`` instead of ``discord.py``, e.g. :: |
||||
|
|
||||
|
python3 -m pip install -U discord.py[voice] |
||||
|
|
||||
|
On Linux environments, installing voice requires getting the following dependencies: |
||||
|
|
||||
|
- libffi |
||||
|
- libnacl |
||||
|
- python3-dev |
||||
|
|
||||
|
For a debian-based system, the following command will help get those dependencies: |
||||
|
|
||||
|
.. code-block:: shell |
||||
|
|
||||
|
$ apt install libffi-dev libnacl-dev python3-dev |
||||
|
|
||||
|
Remember to check your permissions! |
||||
|
|
||||
|
Virtual Environments |
||||
|
~~~~~~~~~~~~~~~~~~~~~ |
||||
|
|
||||
|
Sometimes we don't want to pollute our system installs with a library or we want to maintain |
||||
|
different versions of a library than the currently system installed one. Or we don't have permissions to |
||||
|
install a library along side with the system installed ones. For this purpose, the standard library as |
||||
|
of 3.3 comes with a concept called "Virtual Environment" to help maintain these separate versions. |
||||
|
|
||||
|
A more in-depth tutorial is found on `the official documentation. <https://docs.python.org/3/tutorial/venv.html>`_ |
||||
|
|
||||
|
However, for the quick and dirty: |
||||
|
|
||||
|
1. Go to your project's working directory: |
||||
|
|
||||
|
.. code-block:: shell |
||||
|
|
||||
|
$ cd your-bot-source |
||||
|
$ python3 -m venv bot-env |
||||
|
|
||||
|
2. Activate the virtual environment: |
||||
|
|
||||
|
.. code-block:: shell |
||||
|
|
||||
|
$ source bot-env/bin/activate |
||||
|
|
||||
|
On Windows you activate it with: |
||||
|
|
||||
|
.. code-block:: shell |
||||
|
|
||||
|
$ bot-env\Scripts\activate.bat |
||||
|
|
||||
|
3. Use pip like usual: |
||||
|
|
||||
|
.. code-block:: shell |
||||
|
|
||||
|
$ pip install -U discord.py |
||||
|
|
||||
|
Congratulations. You now have a virtual environment all set up without messing with your system installation. |
||||
|
|
||||
|
Basic Concepts |
||||
|
--------------- |
||||
|
|
||||
|
discord.py revolves around the concept of :ref:`events <discord-api-events>`. |
||||
|
An event is something you listen to and then respond to. For example, when a message |
||||
|
happens, you will receive an event about it and you can then respond to it. |
||||
|
|
||||
|
A quick example to showcase how events work: |
||||
|
|
||||
|
.. code-block:: python |
||||
|
|
||||
|
import discord |
||||
|
|
||||
|
class MyClient(discord.Client): |
||||
|
async def on_ready(self): |
||||
|
print('Logged on as {0}!'.format(self.user)) |
||||
|
|
||||
|
async def on_message(self, message): |
||||
|
print('Message from {0.author}: {0.content}'.format(message)) |
||||
|
|
||||
|
client = MyClient() |
||||
|
client.run('my token goes here') |
||||
|
|
@ -0,0 +1,322 @@ |
|||||
|
:orphan: |
||||
|
|
||||
|
.. currentmodule:: discord |
||||
|
|
||||
|
.. _migrating-to-async: |
||||
|
|
||||
|
Migrating to v0.10.0 |
||||
|
====================== |
||||
|
|
||||
|
v0.10.0 is one of the biggest breaking changes in the library due to massive |
||||
|
fundamental changes in how the library operates. |
||||
|
|
||||
|
The biggest major change is that the library has dropped support to all versions prior to |
||||
|
Python 3.4.2. This was made to support ``asyncio``, in which more detail can be seen |
||||
|
:issue:`in the corresponding issue <50>`. To reiterate this, the implication is that |
||||
|
**python version 2.7 and 3.3 are no longer supported**. |
||||
|
|
||||
|
Below are all the other major changes from v0.9.0 to v0.10.0. |
||||
|
|
||||
|
Event Registration |
||||
|
-------------------- |
||||
|
|
||||
|
All events before were registered using :meth:`Client.event`. While this is still |
||||
|
possible, the events must be decorated with ``@asyncio.coroutine``. |
||||
|
|
||||
|
Before: |
||||
|
|
||||
|
.. code-block:: python |
||||
|
|
||||
|
@client.event |
||||
|
def on_message(message): |
||||
|
pass |
||||
|
|
||||
|
After: |
||||
|
|
||||
|
.. code-block:: python |
||||
|
|
||||
|
@client.event |
||||
|
@asyncio.coroutine |
||||
|
def on_message(message): |
||||
|
pass |
||||
|
|
||||
|
Or in Python 3.5+: |
||||
|
|
||||
|
.. code-block:: python |
||||
|
|
||||
|
@client.event |
||||
|
async def on_message(message): |
||||
|
pass |
||||
|
|
||||
|
Because there is a lot of typing, a utility decorator (:meth:`Client.async_event`) is provided |
||||
|
for easier registration. For example: |
||||
|
|
||||
|
.. code-block:: python |
||||
|
|
||||
|
@client.async_event |
||||
|
def on_message(message): |
||||
|
pass |
||||
|
|
||||
|
|
||||
|
Be aware however, that this is still a coroutine and your other functions that are coroutines must |
||||
|
be decorated with ``@asyncio.coroutine`` or be ``async def``. |
||||
|
|
||||
|
Event Changes |
||||
|
-------------- |
||||
|
|
||||
|
Some events in v0.9.0 were considered pretty useless due to having no separate states. The main |
||||
|
events that were changed were the ``_update`` events since previously they had no context on what |
||||
|
was changed. |
||||
|
|
||||
|
Before: |
||||
|
|
||||
|
.. code-block:: python |
||||
|
|
||||
|
def on_channel_update(channel): pass |
||||
|
def on_member_update(member): pass |
||||
|
def on_status(member): pass |
||||
|
def on_server_role_update(role): pass |
||||
|
def on_voice_state_update(member): pass |
||||
|
def on_socket_raw_send(payload, is_binary): pass |
||||
|
|
||||
|
|
||||
|
After: |
||||
|
|
||||
|
.. code-block:: python |
||||
|
|
||||
|
def on_channel_update(before, after): pass |
||||
|
def on_member_update(before, after): pass |
||||
|
def on_server_role_update(before, after): pass |
||||
|
def on_voice_state_update(before, after): pass |
||||
|
def on_socket_raw_send(payload): pass |
||||
|
|
||||
|
Note that ``on_status`` was removed. If you want its functionality, use :func:`on_member_update`. |
||||
|
See :ref:`discord-api-events` for more information. Other removed events include ``on_socket_closed``, ``on_socket_receive``, and ``on_socket_opened``. |
||||
|
|
||||
|
|
||||
|
Coroutines |
||||
|
----------- |
||||
|
|
||||
|
The biggest change that the library went through is that almost every function in :class:`Client` |
||||
|
was changed to be a `coroutine <https://docs.python.org/3/library/asyncio-task.html>`_. Functions |
||||
|
that are marked as a coroutine in the documentation must be awaited from or yielded from in order |
||||
|
for the computation to be done. For example... |
||||
|
|
||||
|
Before: |
||||
|
|
||||
|
.. code-block:: python |
||||
|
|
||||
|
client.send_message(message.channel, 'Hello') |
||||
|
|
||||
|
After: |
||||
|
|
||||
|
.. code-block:: python |
||||
|
|
||||
|
yield from client.send_message(message.channel, 'Hello') |
||||
|
|
||||
|
# or in python 3.5+ |
||||
|
await client.send_message(message.channel, 'Hello') |
||||
|
|
||||
|
In order for you to ``yield from`` or ``await`` a coroutine then your function must be decorated |
||||
|
with ``@asyncio.coroutine`` or ``async def``. |
||||
|
|
||||
|
Iterables |
||||
|
---------- |
||||
|
|
||||
|
For performance reasons, many of the internal data structures were changed into a dictionary to support faster |
||||
|
lookup. As a consequence, this meant that some lists that were exposed via the API have changed into iterables |
||||
|
and not sequences. In short, this means that certain attributes now only support iteration and not any of the |
||||
|
sequence functions. |
||||
|
|
||||
|
The affected attributes are as follows: |
||||
|
|
||||
|
- :attr:`Client.servers` |
||||
|
- :attr:`Client.private_channels` |
||||
|
- :attr:`Server.channels` |
||||
|
- :attr:`Server.members` |
||||
|
|
||||
|
Some examples of previously valid behaviour that is now invalid |
||||
|
|
||||
|
.. code-block:: python |
||||
|
|
||||
|
if client.servers[0].name == "test": |
||||
|
# do something |
||||
|
|
||||
|
Since they are no longer ``list``\s, they no longer support indexing or any operation other than iterating. |
||||
|
In order to get the old behaviour you should explicitly cast it to a list. |
||||
|
|
||||
|
.. code-block:: python |
||||
|
|
||||
|
servers = list(client.servers) |
||||
|
# work with servers |
||||
|
|
||||
|
.. warning:: |
||||
|
|
||||
|
Due to internal changes of the structure, the order you receive the data in |
||||
|
is not in a guaranteed order. |
||||
|
|
||||
|
Enumerations |
||||
|
------------ |
||||
|
|
||||
|
Due to dropping support for versions lower than Python 3.4.2, the library can now use |
||||
|
`enumerations <https://docs.python.org/3/library/enum.html>`_ in places where it makes sense. |
||||
|
|
||||
|
The common places where this was changed was in the server region, member status, and channel type. |
||||
|
|
||||
|
Before: |
||||
|
|
||||
|
.. code-block:: python |
||||
|
|
||||
|
server.region == 'us-west' |
||||
|
member.status == 'online' |
||||
|
channel.type == 'text' |
||||
|
|
||||
|
After: |
||||
|
|
||||
|
.. code-block:: python |
||||
|
|
||||
|
server.region == discord.ServerRegion.us_west |
||||
|
member.status = discord.Status.online |
||||
|
channel.type == discord.ChannelType.text |
||||
|
|
||||
|
The main reason for this change was to reduce the use of finicky strings in the API as this |
||||
|
could give users a false sense of power. More information can be found in the :ref:`discord-api-enums` page. |
||||
|
|
||||
|
Properties |
||||
|
----------- |
||||
|
|
||||
|
A lot of function calls that returned constant values were changed into Python properties for ease of use |
||||
|
in format strings. |
||||
|
|
||||
|
The following functions were changed into properties: |
||||
|
|
||||
|
+----------------------------------------+--------------------------------------+ |
||||
|
| Before | After | |
||||
|
+----------------------------------------+--------------------------------------+ |
||||
|
| ``User.avatar_url()`` | :attr:`User.avatar_url` | |
||||
|
+----------------------------------------+--------------------------------------+ |
||||
|
| ``User.mention()`` | :attr:`User.mention` | |
||||
|
+----------------------------------------+--------------------------------------+ |
||||
|
| ``Channel.mention()`` | :attr:`Channel.mention` | |
||||
|
+----------------------------------------+--------------------------------------+ |
||||
|
| ``Channel.is_default_channel()`` | :attr:`Channel.is_default` | |
||||
|
+----------------------------------------+--------------------------------------+ |
||||
|
| ``Role.is_everyone()`` | :attr:`Role.is_everyone` | |
||||
|
+----------------------------------------+--------------------------------------+ |
||||
|
| ``Server.get_default_role()`` | :attr:`Server.default_role` | |
||||
|
+----------------------------------------+--------------------------------------+ |
||||
|
| ``Server.icon_url()`` | :attr:`Server.icon_url` | |
||||
|
+----------------------------------------+--------------------------------------+ |
||||
|
| ``Server.get_default_channel()`` | :attr:`Server.default_channel` | |
||||
|
+----------------------------------------+--------------------------------------+ |
||||
|
| ``Message.get_raw_mentions()`` | :attr:`Message.raw_mentions` | |
||||
|
+----------------------------------------+--------------------------------------+ |
||||
|
| ``Message.get_raw_channel_mentions()`` | :attr:`Message.raw_channel_mentions` | |
||||
|
+----------------------------------------+--------------------------------------+ |
||||
|
|
||||
|
Member Management |
||||
|
------------------- |
||||
|
|
||||
|
Functions that involved banning and kicking were changed. |
||||
|
|
||||
|
+--------------------------------+--------------------------+ |
||||
|
| Before | After | |
||||
|
+--------------------------------+--------------------------+ |
||||
|
| ``Client.ban(server, user)`` | ``Client.ban(member)`` | |
||||
|
+--------------------------------+--------------------------+ |
||||
|
| ``Client.kick(server, user)`` | ``Client.kick(member)`` | |
||||
|
+--------------------------------+--------------------------+ |
||||
|
|
||||
|
.. migrating-renames: |
||||
|
|
||||
|
Renamed Functions |
||||
|
------------------- |
||||
|
|
||||
|
Functions have been renamed. |
||||
|
|
||||
|
+------------------------------------+-------------------------------------------+ |
||||
|
| Before | After | |
||||
|
+------------------------------------+-------------------------------------------+ |
||||
|
| ``Client.set_channel_permissions`` | :meth:`Client.edit_channel_permissions` | |
||||
|
+------------------------------------+-------------------------------------------+ |
||||
|
|
||||
|
All the :class:`Permissions` related attributes have been renamed and the `can_` prefix has been |
||||
|
dropped. So for example, ``can_manage_messages`` has become ``manage_messages``. |
||||
|
|
||||
|
Forced Keyword Arguments |
||||
|
------------------------- |
||||
|
|
||||
|
Since 3.0+ of Python, we can now force questions to take in forced keyword arguments. A keyword argument is when you |
||||
|
explicitly specify the name of the variable and assign to it, for example: ``foo(name='test')``. Due to this support, |
||||
|
some functions in the library were changed to force things to take said keyword arguments. This is to reduce errors of |
||||
|
knowing the argument order and the issues that could arise from them. |
||||
|
|
||||
|
The following parameters are now exclusively keyword arguments: |
||||
|
|
||||
|
- :meth:`Client.send_message` |
||||
|
- ``tts`` |
||||
|
- :meth:`Client.logs_from` |
||||
|
- ``before`` |
||||
|
- ``after`` |
||||
|
- :meth:`Client.edit_channel_permissions` |
||||
|
- ``allow`` |
||||
|
- ``deny`` |
||||
|
|
||||
|
In the documentation you can tell if a function parameter is a forced keyword argument if it is after ``\*,`` |
||||
|
in the function signature. |
||||
|
|
||||
|
.. _migrating-running: |
||||
|
|
||||
|
Running the Client |
||||
|
-------------------- |
||||
|
|
||||
|
In earlier versions of discord.py, ``client.run()`` was a blocking call to the main thread |
||||
|
that called it. In v0.10.0 it is still a blocking call but it handles the event loop for you. |
||||
|
However, in order to do that you must pass in your credentials to :meth:`Client.run`. |
||||
|
|
||||
|
Basically, before: |
||||
|
|
||||
|
.. code-block:: python |
||||
|
|
||||
|
client.login('token') |
||||
|
client.run() |
||||
|
|
||||
|
After: |
||||
|
|
||||
|
.. code-block:: python |
||||
|
|
||||
|
client.run('token') |
||||
|
|
||||
|
.. warning:: |
||||
|
|
||||
|
Like in the older ``Client.run`` function, the newer one must be the one of |
||||
|
the last functions to call. This is because the function is **blocking**. Registering |
||||
|
events or doing anything after :meth:`Client.run` will not execute until the function |
||||
|
returns. |
||||
|
|
||||
|
This is a utility function that abstracts the event loop for you. There's no need for |
||||
|
the run call to be blocking and out of your control. Indeed, if you want control of the |
||||
|
event loop then doing so is quite straightforward: |
||||
|
|
||||
|
.. code-block:: python |
||||
|
|
||||
|
import discord |
||||
|
import asyncio |
||||
|
|
||||
|
client = discord.Client() |
||||
|
|
||||
|
@asyncio.coroutine |
||||
|
def main_task(): |
||||
|
yield from client.login('token') |
||||
|
yield from client.connect() |
||||
|
|
||||
|
loop = asyncio.get_event_loop() |
||||
|
try: |
||||
|
loop.run_until_complete(main_task()) |
||||
|
except: |
||||
|
loop.run_until_complete(client.logout()) |
||||
|
finally: |
||||
|
loop.close() |
||||
|
|
||||
|
|
||||
|
|
@ -0,0 +1,76 @@ |
|||||
|
.. _quickstart: |
||||
|
|
||||
|
.. currentmodule:: discord |
||||
|
|
||||
|
Quickstart |
||||
|
============ |
||||
|
|
||||
|
This page gives a brief introduction to the library. It assumes you have the library installed, |
||||
|
if you don't check the :ref:`installing` portion. |
||||
|
|
||||
|
A Minimal Bot |
||||
|
--------------- |
||||
|
|
||||
|
Let's make a bot that replies to a specific message and walk you through it. |
||||
|
|
||||
|
It looks something like this: |
||||
|
|
||||
|
.. code-block:: python |
||||
|
|
||||
|
import discord |
||||
|
|
||||
|
client = discord.Client() |
||||
|
|
||||
|
@client.event |
||||
|
async def on_ready(): |
||||
|
print('We have logged in as {0.user}'.format(self)) |
||||
|
|
||||
|
@client.event |
||||
|
async def on_message(message): |
||||
|
if message.author == client.user: |
||||
|
return |
||||
|
|
||||
|
if message.content.startswith('$hello'): |
||||
|
await message.channel.send('Hello!') |
||||
|
|
||||
|
client.run('your token here') |
||||
|
|
||||
|
Let's name this file ``example_bot.py``. Make sure not to name it ``discord.py`` as that'll conflict |
||||
|
with the library. |
||||
|
|
||||
|
There's a lot going on here, so let's walk you through it step by step. |
||||
|
|
||||
|
1. The first line just imports the library, if this raises a `ModuleNotFoundError` or `ImportError` |
||||
|
then head on over to :ref:`installing` section to properly install. |
||||
|
2. Next, we create an instance of a :class:`Client`. This client is our connection to Discord. |
||||
|
3. We then use the :meth:`Client.event` decorator to register an event. This library has many events. |
||||
|
Since this library is asynchronous, we do things in a "callback" style manner. |
||||
|
|
||||
|
A callback is essentially a function that is called when something happens. In our case, |
||||
|
the :func:`on_ready` event is called when the bot has finished logging in and setting things |
||||
|
up and the :func:`on_message` event is called when the bot has received a message. |
||||
|
4. Since the :func:`on_message` event triggers for *every* message received, we have to make |
||||
|
sure that we ignore messages from ourselves. We do this by checking if the :attr:`Message.author` |
||||
|
is the same as the :attr:`Client.user`. |
||||
|
5. Afterwards, we check if the :class:`Message.content` starts with ``'$hello'``. If it is, |
||||
|
then we reply in the channel it was used in with ``'Hello!'``. |
||||
|
6. Finally, we run the bot with our login token. If you need help getting your token or creating a bot, |
||||
|
look in the :ref:`discord-intro` section. |
||||
|
|
||||
|
|
||||
|
Now that we've made a bot, we have to *run* the bot. Luckily, this is simple since this is just a |
||||
|
Python script, we can run it directly. |
||||
|
|
||||
|
On Windows: |
||||
|
|
||||
|
.. code-block:: shell |
||||
|
|
||||
|
$ py -3 example_bot.py |
||||
|
|
||||
|
On other systems: |
||||
|
|
||||
|
.. code-block:: shell |
||||
|
|
||||
|
$ python3 example_bot.py |
||||
|
|
||||
|
Now you can try playing around with your basic bot. |