Захотелось как-то мне, чтобы сообщения одного из чатов телеграма сохранялись у меня на диске (не запуская обычного клиента). Не буду раскрывать своих побудительных мотивов, но возможность эта показалась мне нужной и полезной.


Для этого в телеграме есть боты. На Хабре есть несколько статей, посвященных ботам, например: "Чат-помощник на сайт".


Бот позволяет читать и посылать сообщения, для регистрации бота не нужен телефон и количество ботов может быть любым. Но название бота включает в себя слово "bot", что может вызвать у хозяина чата ненужные вопросы.


Но, как говорится, правильно поставленный вопрос — половина ответа.


Оказывается кроме "API telegram bot" существует еще и "API telegram client", т.е. API для создания собственных клиентов.


Клиент также может посылать и читать сообщения, но только от зарегестрированного (привязанного к телефону) пользователя, что мне как раз подходит (я уже зарегестрирован в чате).


На сайте телеграма есть список API для разных платформ: https://telegram.org/apps#source-code


Однако, самой простой в использовании оказалась библиотека на python: Pure Python 3 MTProto API Telegram client library под названием "telethon"


Только вот проблема. Я не знаю питон. Ну что ж, есть повод познакомиться.
Как утверждает мануал по телетону, инсталляция его очень простая. Достаточно запустить команду в командной строке:


pip3 install telethon

Подводные камни, встреченные мною при инсталляции:


  • не инсталлирован pip3 (инсталлятор для питона).

sudo apt-get -y install python3-pip


  • Библиотека работает только на питоне версии >3.5. Так что, возможно, придется его обновить.

Все установилось. Листаем readme.txt дальше.


Следущим пунктом идет создание клиента телеграма… Как, уже? Ну да, все просто. Правда, сперва нужно зарегистритровать себя как создателя клиента.


Заходим на сайт телеграма: https://my.telegram.org
Вводим телефон и ждем код подтверждения на родном клиенте телеграма. Он довольно длинный (12 символов) и неудобный для ввода.


Заходим в пункт "API". Ищем "Telegram API" и заходим в "Creating an application" (https://my.telegram.org/apps).


Заполняем поля App title и Short name, нажимаем «Create application» и запоминаем две переменные: api_id и api_hash.


Пришла пора делать клиента.


from telethon import TelegramClient, sync

# Вставляем api_id и api_hash
api_id = 12345
api_hash = '0123456789abcdef0123456789abcdef'

client = TelegramClient('session_name', api_id, api_hash)
client.start()

session_name — можно вставить любое имя. Вас попросят ввести телефон и пришлют код подтверждения. После этого клиент будет работать без запроса телефона (до тех пор, пока не поменяете session_name). Рядом с вашей программой появится файл session_name.session


Если ошибок нет, клиент готов. Только вот, ничего не выводит. Попробуем получить полезную инфорфмацию.


Узнаем немного о себе:


print(client.get_me().stringify())

результат выдается в виде:


User(
    photo=None,
    last_name='Pupkin',
    first_name='Vasya',
    id=123456789,
    phone='79041234567',
.... - что-то еще...
)

Можем послать сообщение от себя:


client.send_message('username', 'Hello! Talking to you from Telethon')

Можно и картинку


client.send_file('username', '/home/myself/Pictures/holidays.jpg')

Как меня видят другие:


client.download_profile_photo('me')

Смотрим, на какие чаты мы подписаны:


print all chats name
for dialog in client.iter_dialogs():
    print(dialog.title)

читаем все сообщения чата "chat_name" (осторожно, сообщений может быть очень много)


messages = client.get_entity('chat_name')
print(messages)

просмотр всех пользователей чата


participants = client.get_participants('chat_name')
print(participants)

Побаловались?
Теперь, собственно, делаем то, ради чего мы все это затеяли...


Нам нужна программка, следящая за новыми сообщениями в определенном канале.


Чтобы клиент не заканчивал работу, после client.start() вставляем строку:


client.run_until_disconnected()

Эта конструкция (вставляется перед client.start()) выводит только новые сообщения:


@client.on(events.NewMessage(chats=('chat_name')))
async def normal_handler(event):
#    print(event.message)
    print(event.message.to_dict()['message'])

Давайте разберемся.


@client.on(events.NewMessage(chats=('chat_name')))

создает событие, срабатывающее при появлении нового сообщения


 print(event.message)

выводит сообщение в таком виде:


Message(edit_date=None, views=None, reply_markup=None, fwd_from=None, id=187, entities=[], post=False, mentioned=False, via_bot_id=None, media_unread=False, out=True, media=None, date=datetime.datetime(2018, 10, 1, 9, 26, 21, tzinfo=datetime.timezone.utc), to_id=PeerChannel(channel_id=123456789), reply_to_msg_id=None, from_id=123456789, silent=False, grouped_id=None, post_author=None, message='hello telegram')

Из всего этого нам нужно поле: "message='hello telegram'":


    print(event.message.to_dict()['message'])

Сообщение получили, но от кого оно, непонятно, т.к. в сообщение только ID пользователя. Чтобы сопоставить ID и имя пользователя, скачиваем всех пользователей чата и помещаем их в словарь (хэш) в виде d[id]="first_name last_name"


participants = client.get_participants(group)
users={}

for partic in client.iter_participants(group):
    lastname=""
    if partic.last_name:
       lastname=partic.last_name
    users[partic.id]=partic.first_name+" "+lastname

Теперь мы можем узнать, кто послал сообщение:


s_user_id=event.message.to_dict()['from_id']
user_id=int(s_user_id)
user=d.get(user_id)

В принципе, можно получить имя пользователя из телеграма напрямую, но если пользователей немного, проще со словарем.


Вытаскиваем из сообщения дату отправки:


mess_date=event.message.to_dict()['date']

Все, все данные у нас есть. Осталось записать их в файл.
Для этого сначала откроем файл на запись:


f=open('messages_from_chat', 'a') 

И запишем сообщение:


f.write(mess_date.strftime("%d-%m-%Y %H:%M")+"\n")
f.write(user+"\n")
f.write(user_mess+"\n\n")
f.flush()

Вот и все! Все, что мне было нужно, программка делает. Утилитка, конечно, сыровата, но свою задачу выполняет.


Python оказался не таким уж и сложным как его малюют, тем более описание и разных библиотек в интернете полным-полно. Написать еще пару утилиток и привыкнув к нему, можно использовать его как скриптовый язык вместо bash.


Весь текст утилиты:
from telethon import TelegramClient, sync, events

api_id = 12345
api_hash = '0123456789abcdef0123456789abcdef'

client = TelegramClient('session_name', api_id, api_hash)

@client.on(events.NewMessage(chats=('chat_name')))
async def normal_handler(event):
#    print(event.message)
    user_mess=event.message.to_dict()['message']

    s_user_id=event.message.to_dict()['from_id']
    user_id=int(s_user_id)
    user=d.get(user_id)

    mess_date=event.message.to_dict()['date']

    f.write(mess_date.strftime("%d-%m-%Y %H:%M")+"\n")
    f.write(user+"\n")
    f.write(user_mess+"\n\n")

    f.flush()

client.start()

group='group_name'

participants = client.get_participants(group)
users={}

for partic in client.iter_participants(group):
    lastname=""
    if partic.last_name:
       lastname=partic.last_name
    users[partic.id]=partic.first_name+" "+lastname

f=open('messages_from_chat', 'a') 

client.run_until_disconnected()
f.close()

Полное описание библиотеки.

Комментарии (5)


  1. munrocket
    02.10.2018 19:12

    Скажите, а telethon тянет стабильно сообщения 24/7? Я правильно понимаю что вы подписались на update event в канале? По сравнению с С# клиентской либой у telethon на много удобнее интерфейс.


    1. prohfessor Автор
      03.10.2018 05:52

      На 24/7 еще не проверял. Время покажет :-).
      Вот такой конструкцией я «подписываюсь» на новые сообщения: "@client.on(events.NewMessage(chats=('chat_name')))."
      А это callback функция, вызываемая при приходе нового сообщения: «async def normal_handler(event):».


    1. Anastasia_K
      03.10.2018 10:21

      да, телетон стабильно работает 24/7. может вывалиться конечно при проблемах на серверах телеграмма, но случается это довольно редко


  1. Zanak
    03.10.2018 19:22

    «Подслушиваем чат...» — ожидал чего — то, если не о взломе, то о хитрых приемах получения от телеграмма больше информации, чем этого предусмотрели создатели, а здесь, всего лишь, рассказ о питонячей версии клиента.
    Хотя сама информация интересна, а временами может оказаться и востребованной, я все таки немного разочарован :)


  1. ukunti
    04.10.2018 05:03

    Очередной глупейший кликбейт от нуба про телеграм…