2018-01-05 02:59:53 +03:00
|
|
|
===============================
|
|
|
|
Working with Chats and Channels
|
|
|
|
===============================
|
|
|
|
|
|
|
|
|
2018-01-20 13:47:17 +03:00
|
|
|
.. note::
|
|
|
|
|
|
|
|
These examples assume you have read :ref:`accessing-the-full-api`.
|
|
|
|
|
|
|
|
|
2018-01-05 02:59:53 +03:00
|
|
|
Joining a chat or channel
|
|
|
|
*************************
|
|
|
|
|
2018-03-27 19:22:01 +03:00
|
|
|
Note that :tl:`Chat` are normal groups, and :tl:`Channel` are a
|
2018-06-22 15:44:59 +03:00
|
|
|
special form of :tl:`Chat`, which can also be super-groups if
|
2018-03-27 19:22:01 +03:00
|
|
|
their ``megagroup`` member is ``True``.
|
2018-01-05 02:59:53 +03:00
|
|
|
|
|
|
|
|
|
|
|
Joining a public channel
|
|
|
|
************************
|
|
|
|
|
|
|
|
Once you have the :ref:`entity <entities>` of the channel you want to join
|
2018-04-08 15:15:26 +03:00
|
|
|
to, you can make use of the :tl:`JoinChannelRequest` to join such channel:
|
2018-01-05 02:59:53 +03:00
|
|
|
|
2018-06-20 12:05:33 +03:00
|
|
|
.. code-block:: python
|
2018-01-05 02:59:53 +03:00
|
|
|
|
2018-06-20 12:05:33 +03:00
|
|
|
from telethon.tl.functions.channels import JoinChannelRequest
|
2018-06-22 15:44:59 +03:00
|
|
|
loop.run_until_complete(client(JoinChannelRequest(channel)))
|
2018-01-05 02:59:53 +03:00
|
|
|
|
2018-06-20 12:05:33 +03:00
|
|
|
# In the same way, you can also leave such channel
|
|
|
|
from telethon.tl.functions.channels import LeaveChannelRequest
|
2018-06-22 15:44:59 +03:00
|
|
|
loop.run_until_complete(client(LeaveChannelRequest(input_channel)))
|
2018-01-05 02:59:53 +03:00
|
|
|
|
|
|
|
|
|
|
|
For more on channels, check the `channels namespace`__.
|
|
|
|
|
|
|
|
|
2018-04-06 20:00:21 +03:00
|
|
|
__ https://lonamiwebs.github.io/Telethon/methods/channels/index.html
|
|
|
|
|
|
|
|
|
2018-01-05 02:59:53 +03:00
|
|
|
Joining a private chat or channel
|
|
|
|
*********************************
|
|
|
|
|
|
|
|
If all you have is a link like this one:
|
|
|
|
``https://t.me/joinchat/AAAAAFFszQPyPEZ7wgxLtd``, you already have
|
|
|
|
enough information to join! The part after the
|
|
|
|
``https://t.me/joinchat/``, this is, ``AAAAAFFszQPyPEZ7wgxLtd`` on this
|
|
|
|
example, is the ``hash`` of the chat or channel. Now you can use
|
2018-04-08 15:15:26 +03:00
|
|
|
:tl:`ImportChatInviteRequest` as follows:
|
2018-01-05 02:59:53 +03:00
|
|
|
|
2018-06-20 12:05:33 +03:00
|
|
|
.. code-block:: python
|
2018-01-05 02:59:53 +03:00
|
|
|
|
2018-06-20 12:05:33 +03:00
|
|
|
from telethon.tl.functions.messages import ImportChatInviteRequest
|
2018-06-22 15:44:59 +03:00
|
|
|
updates = loop.run_until_complete(
|
|
|
|
client(ImportChatInviteRequest('AAAAAEHbEkejzxUjAUCfYg'))
|
|
|
|
)
|
2018-01-05 02:59:53 +03:00
|
|
|
|
|
|
|
|
|
|
|
Adding someone else to such chat or channel
|
|
|
|
*******************************************
|
|
|
|
|
|
|
|
If you don't want to add yourself, maybe because you're already in,
|
2018-04-08 15:15:26 +03:00
|
|
|
you can always add someone else with the :tl:`AddChatUserRequest`, which
|
|
|
|
use is very straightforward, or :tl:`InviteToChannelRequest` for channels:
|
2018-01-05 02:59:53 +03:00
|
|
|
|
2018-06-20 12:05:33 +03:00
|
|
|
.. code-block:: python
|
2018-01-05 02:59:53 +03:00
|
|
|
|
2018-06-20 12:05:33 +03:00
|
|
|
# For normal chats
|
|
|
|
from telethon.tl.functions.messages import AddChatUserRequest
|
2018-01-05 02:59:53 +03:00
|
|
|
|
2018-06-20 12:05:33 +03:00
|
|
|
# Note that ``user_to_add`` is NOT the name of the parameter.
|
|
|
|
# It's the user you want to add (``user_id=user_to_add``).
|
2018-06-22 15:44:59 +03:00
|
|
|
loop.run_until_complete(client(AddChatUserRequest(
|
2018-06-20 12:05:33 +03:00
|
|
|
chat_id,
|
|
|
|
user_to_add,
|
|
|
|
fwd_limit=10 # Allow the user to see the 10 last messages
|
2018-06-22 15:44:59 +03:00
|
|
|
)))
|
2018-01-05 02:59:53 +03:00
|
|
|
|
2018-06-20 12:05:33 +03:00
|
|
|
# For channels (which includes megagroups)
|
|
|
|
from telethon.tl.functions.channels import InviteToChannelRequest
|
2018-03-22 20:39:42 +03:00
|
|
|
|
2018-06-22 15:44:59 +03:00
|
|
|
loop.run_until_complete(client(InviteToChannelRequest(
|
2018-06-20 12:05:33 +03:00
|
|
|
channel,
|
|
|
|
[users_to_add]
|
2018-06-22 15:44:59 +03:00
|
|
|
)))
|
2018-03-22 20:39:42 +03:00
|
|
|
|
|
|
|
|
2018-01-05 02:59:53 +03:00
|
|
|
Checking a link without joining
|
|
|
|
*******************************
|
|
|
|
|
|
|
|
If you don't need to join but rather check whether it's a group or a
|
2018-04-08 15:15:26 +03:00
|
|
|
channel, you can use the :tl:`CheckChatInviteRequest`, which takes in
|
2018-01-08 16:04:04 +03:00
|
|
|
the hash of said channel or group.
|
2018-01-05 02:59:53 +03:00
|
|
|
|
2018-04-06 20:00:21 +03:00
|
|
|
|
2018-01-05 02:59:53 +03:00
|
|
|
Retrieving all chat members (channels too)
|
|
|
|
******************************************
|
|
|
|
|
2018-05-17 13:00:22 +03:00
|
|
|
.. note::
|
|
|
|
|
|
|
|
Use the `telethon.telegram_client.TelegramClient.iter_participants`
|
|
|
|
friendly method instead unless you have a better reason not to!
|
|
|
|
|
|
|
|
This method will handle different chat types for you automatically.
|
2018-03-27 19:22:01 +03:00
|
|
|
|
|
|
|
|
2018-06-22 15:44:59 +03:00
|
|
|
Here is the easy way to do it:
|
|
|
|
|
|
|
|
.. code-block:: python
|
|
|
|
|
|
|
|
participants = loop.run_until_complete(client.get_participants(group))
|
|
|
|
|
|
|
|
Now we will show how the method works internally.
|
|
|
|
|
2018-01-05 02:59:53 +03:00
|
|
|
In order to get all the members from a mega-group or channel, you need
|
2018-04-08 15:15:26 +03:00
|
|
|
to use :tl:`GetParticipantsRequest`. As we can see it needs an
|
|
|
|
:tl:`InputChannel`, (passing the mega-group or channel you're going to
|
|
|
|
use will work), and a mandatory :tl:`ChannelParticipantsFilter`. The
|
2018-01-05 02:59:53 +03:00
|
|
|
closest thing to "no filter" is to simply use
|
2018-04-08 15:15:26 +03:00
|
|
|
:tl:`ChannelParticipantsSearch` with an empty ``'q'`` string.
|
2018-01-05 02:59:53 +03:00
|
|
|
|
|
|
|
If we want to get *all* the members, we need to use a moving offset and
|
|
|
|
a fixed limit:
|
|
|
|
|
2018-06-20 12:05:33 +03:00
|
|
|
.. code-block:: python
|
2018-01-05 02:59:53 +03:00
|
|
|
|
2018-06-20 12:05:33 +03:00
|
|
|
from telethon.tl.functions.channels import GetParticipantsRequest
|
|
|
|
from telethon.tl.types import ChannelParticipantsSearch
|
|
|
|
from time import sleep
|
2018-01-05 02:59:53 +03:00
|
|
|
|
2018-06-20 12:05:33 +03:00
|
|
|
offset = 0
|
|
|
|
limit = 100
|
|
|
|
all_participants = []
|
2018-01-05 02:59:53 +03:00
|
|
|
|
2018-06-20 12:05:33 +03:00
|
|
|
while True:
|
2018-06-22 15:44:59 +03:00
|
|
|
participants = loop.run_until_complete(client(GetParticipantsRequest(
|
|
|
|
channel, ChannelParticipantsSearch(''), offset, limit, hash=0
|
|
|
|
)))
|
2018-06-20 12:05:33 +03:00
|
|
|
if not participants.users:
|
|
|
|
break
|
|
|
|
all_participants.extend(participants.users)
|
|
|
|
offset += len(participants.users)
|
2018-01-05 02:59:53 +03:00
|
|
|
|
|
|
|
|
2018-03-01 15:21:28 +03:00
|
|
|
.. note::
|
|
|
|
|
2018-03-27 19:22:01 +03:00
|
|
|
If you need more than 10,000 members from a group you should use the
|
|
|
|
mentioned ``client.get_participants(..., aggressive=True)``. It will
|
|
|
|
do some tricks behind the scenes to get as many entities as possible.
|
|
|
|
Refer to `issue 573`__ for more on this.
|
2018-03-01 15:21:28 +03:00
|
|
|
|
|
|
|
|
2018-04-08 15:15:26 +03:00
|
|
|
Note that :tl:`GetParticipantsRequest` returns :tl:`ChannelParticipants`,
|
2018-01-05 02:59:53 +03:00
|
|
|
which may have more information you need (like the role of the
|
|
|
|
participants, total count of members, etc.)
|
|
|
|
|
2018-03-01 15:21:28 +03:00
|
|
|
__ https://github.com/LonamiWebs/Telethon/issues/573
|
2018-01-05 02:59:53 +03:00
|
|
|
|
|
|
|
|
|
|
|
Recent Actions
|
|
|
|
**************
|
|
|
|
|
|
|
|
"Recent actions" is simply the name official applications have given to
|
2018-04-08 15:15:26 +03:00
|
|
|
the "admin log". Simply use :tl:`GetAdminLogRequest` for that, and
|
2018-01-05 02:59:53 +03:00
|
|
|
you'll get AdminLogResults.events in return which in turn has the final
|
|
|
|
`.action`__.
|
|
|
|
|
|
|
|
__ https://lonamiwebs.github.io/Telethon/types/channel_admin_log_event_action.html
|
|
|
|
|
|
|
|
|
|
|
|
Admin Permissions
|
|
|
|
*****************
|
|
|
|
|
2018-04-08 15:15:26 +03:00
|
|
|
Giving or revoking admin permissions can be done with the :tl:`EditAdminRequest`:
|
2018-01-05 02:59:53 +03:00
|
|
|
|
2018-06-20 12:05:33 +03:00
|
|
|
.. code-block:: python
|
|
|
|
|
|
|
|
from telethon.tl.functions.channels import EditAdminRequest
|
|
|
|
from telethon.tl.types import ChannelAdminRights
|
|
|
|
|
|
|
|
# You need both the channel and who to grant permissions
|
|
|
|
# They can either be channel/user or input channel/input user.
|
|
|
|
#
|
|
|
|
# ChannelAdminRights is a list of granted permissions.
|
|
|
|
# Set to True those you want to give.
|
|
|
|
rights = ChannelAdminRights(
|
|
|
|
post_messages=None,
|
|
|
|
add_admins=None,
|
|
|
|
invite_users=None,
|
|
|
|
change_info=True,
|
|
|
|
ban_users=None,
|
|
|
|
delete_messages=True,
|
|
|
|
pin_messages=True,
|
|
|
|
invite_link=None,
|
|
|
|
edit_messages=None
|
|
|
|
)
|
|
|
|
# Equivalent to:
|
|
|
|
# rights = ChannelAdminRights(
|
|
|
|
# change_info=True,
|
|
|
|
# delete_messages=True,
|
|
|
|
# pin_messages=True
|
|
|
|
# )
|
|
|
|
|
|
|
|
# Once you have a ChannelAdminRights, invoke it
|
2018-06-22 15:44:59 +03:00
|
|
|
loop.run_until_complete(client(EditAdminRequest(channel, user, rights)))
|
2018-06-20 12:05:33 +03:00
|
|
|
|
|
|
|
# User will now be able to change group info, delete other people's
|
|
|
|
# messages and pin messages.
|
2018-04-06 20:00:21 +03:00
|
|
|
|
|
|
|
|
|
|
|
.. note::
|
|
|
|
|
|
|
|
Thanks to `@Kyle2142`__ for `pointing out`__ that you **cannot** set all
|
|
|
|
parameters to ``True`` to give a user full permissions, as not all
|
|
|
|
permissions are related to both broadcast channels/megagroups.
|
|
|
|
|
|
|
|
E.g. trying to set ``post_messages=True`` in a megagroup will raise an
|
|
|
|
error. It is recommended to always use keyword arguments, and to set only
|
|
|
|
the permissions the user needs. If you don't need to change a permission,
|
|
|
|
it can be omitted (full list `here`__).
|
|
|
|
|
|
|
|
|
|
|
|
Restricting Users
|
|
|
|
*****************
|
|
|
|
|
|
|
|
Similar to how you give or revoke admin permissions, you can edit the
|
2018-06-16 22:51:13 +03:00
|
|
|
banned rights of an user through :tl:`EditBannedRequest` and its parameter
|
2018-04-08 15:15:26 +03:00
|
|
|
:tl:`ChannelBannedRights`:
|
2018-04-06 20:00:21 +03:00
|
|
|
|
2018-06-20 12:05:33 +03:00
|
|
|
.. code-block:: python
|
|
|
|
|
|
|
|
from telethon.tl.functions.channels import EditBannedRequest
|
|
|
|
from telethon.tl.types import ChannelBannedRights
|
|
|
|
|
|
|
|
from datetime import datetime, timedelta
|
|
|
|
|
|
|
|
# Restricting an user for 7 days, only allowing view/send messages.
|
|
|
|
#
|
|
|
|
# Note that it's "reversed". You must set to ``True`` the permissions
|
|
|
|
# you want to REMOVE, and leave as ``None`` those you want to KEEP.
|
|
|
|
rights = ChannelBannedRights(
|
|
|
|
until_date=datetime.now() + timedelta(days=7),
|
|
|
|
view_messages=None,
|
|
|
|
send_messages=None,
|
|
|
|
send_media=True,
|
|
|
|
send_stickers=True,
|
|
|
|
send_gifs=True,
|
|
|
|
send_games=True,
|
|
|
|
send_inline=True,
|
|
|
|
embed_links=True
|
|
|
|
)
|
|
|
|
|
|
|
|
# The above is equivalent to
|
|
|
|
rights = ChannelBannedRights(
|
|
|
|
until_date=datetime.now() + timedelta(days=7),
|
|
|
|
send_media=True,
|
|
|
|
send_stickers=True,
|
|
|
|
send_gifs=True,
|
|
|
|
send_games=True,
|
|
|
|
send_inline=True,
|
|
|
|
embed_links=True
|
|
|
|
)
|
|
|
|
|
2018-06-22 15:44:59 +03:00
|
|
|
loop.run_until_complete(client(EditBannedRequest(channel, user, rights)))
|
2018-04-06 20:00:21 +03:00
|
|
|
|
|
|
|
|
|
|
|
Kicking a member
|
|
|
|
****************
|
|
|
|
|
|
|
|
Telegram doesn't actually have a request to kick an user from a group.
|
|
|
|
Instead, you need to restrict them so they can't see messages. Any date
|
|
|
|
is enough:
|
|
|
|
|
2018-06-20 12:05:33 +03:00
|
|
|
.. code-block:: python
|
2018-04-06 20:00:21 +03:00
|
|
|
|
2018-06-20 12:05:33 +03:00
|
|
|
from telethon.tl.functions.channels import EditBannedRequest
|
|
|
|
from telethon.tl.types import ChannelBannedRights
|
2018-04-06 20:00:21 +03:00
|
|
|
|
2018-06-22 15:44:59 +03:00
|
|
|
loop.run_until_complete(client(EditBannedRequest(
|
|
|
|
channel, user, ChannelBannedRights(
|
|
|
|
until_date=None,
|
|
|
|
view_messages=True
|
|
|
|
)
|
2018-06-20 12:05:33 +03:00
|
|
|
)))
|
2018-04-06 20:00:21 +03:00
|
|
|
|
2018-01-05 02:59:53 +03:00
|
|
|
|
|
|
|
__ https://github.com/Kyle2142
|
|
|
|
__ https://github.com/LonamiWebs/Telethon/issues/490
|
|
|
|
__ https://lonamiwebs.github.io/Telethon/constructors/channel_admin_rights.html
|
|
|
|
|
|
|
|
|
|
|
|
Increasing View Count in a Channel
|
|
|
|
**********************************
|
|
|
|
|
|
|
|
It has been asked `quite`__ `a few`__ `times`__ (really, `many`__), and
|
|
|
|
while I don't understand why so many people ask this, the solution is to
|
2018-04-08 15:15:26 +03:00
|
|
|
use :tl:`GetMessagesViewsRequest`, setting ``increment=True``:
|
2018-01-05 02:59:53 +03:00
|
|
|
|
2018-06-20 12:05:33 +03:00
|
|
|
.. code-block:: python
|
2018-01-05 02:59:53 +03:00
|
|
|
|
|
|
|
|
2018-06-20 12:05:33 +03:00
|
|
|
# Obtain `channel' through dialogs or through client.get_entity() or anyhow.
|
|
|
|
# Obtain `msg_ids' through `.get_messages()` or anyhow. Must be a list.
|
2018-01-05 02:59:53 +03:00
|
|
|
|
2018-06-22 15:44:59 +03:00
|
|
|
loop.run_until_complete(client(GetMessagesViewsRequest(
|
2018-06-20 12:05:33 +03:00
|
|
|
peer=channel,
|
|
|
|
id=msg_ids,
|
|
|
|
increment=True
|
2018-06-22 15:44:59 +03:00
|
|
|
)))
|
2018-01-05 02:59:53 +03:00
|
|
|
|
2018-03-22 20:39:42 +03:00
|
|
|
|
|
|
|
Note that you can only do this **once or twice a day** per account,
|
|
|
|
running this in a loop will obviously not increase the views forever
|
|
|
|
unless you wait a day between each iteration. If you run it any sooner
|
|
|
|
than that, the views simply won't be increased.
|
|
|
|
|
2018-01-05 02:59:53 +03:00
|
|
|
__ https://github.com/LonamiWebs/Telethon/issues/233
|
|
|
|
__ https://github.com/LonamiWebs/Telethon/issues/305
|
|
|
|
__ https://github.com/LonamiWebs/Telethon/issues/409
|
|
|
|
__ https://github.com/LonamiWebs/Telethon/issues/447
|