Arrow
ArrowТам. 2, 2019, 7:06 Т.Ж.

Деплой Telegram бота на Django с исползованием webhook

Bot, Django, Telegram

Добрый день.

Помогите пожалуйста никак не могу найти информации о деплое Telegram бота на pythonenywhere.

Написал бота используя pyTelegramBotAPI (база данных, админка и еще некоторые вещи на Django). Столкнулся с проблемой, что на локальной машине с использованием polling() все работает, а когда выкладываю на pythonenywhere, чтобы не мучать сервер нужно использовать webhook. Только нигде в интернете нет вменяемого объяснения как сделать это для Django. Для Flask куча описаний. Пытался переделать то, что удалось найти, но бот никак не заводится. Вернее само приложение Django работает прекрасно, как и админка бота, только бот ника не реагирует на мои сообщения в телеграм. По логам все ок и теграм говорит, что webhook установлен успешно.

Вот мой код:

Файл "urls.py" проекта:

urlpatterns = [
    path('admin/', admin.site.urls),
    path('', include('bot.urls')),
    ..................
]

Файл "urls.py" бота:

urlpatterns = [
    path('', UpdateBot.as_view(), name='update'),
]

Файл "views.py" бота:

from telebot import TeleBot, types
from rest_framework.response import Response
from rest_framework.views import APIView
from TelegramBot.settings import TOKEN


bot = TeleBot(TOKEN)


class UpdateBot(APIView):
    def post(self, request):
        # Сюда должны получать сообщения от телеграм и далее обрабатываться ботом
        json_str = request.body.decode('UTF-8')
        update = types.Update.de_json(json_str)
        bot.process_new_updates([update])

        return Response({'code': 200})


@bot.message_handler(commands=['start'])
def start_message(message):
    # User написал /start в диалоге с ботом
    text = '<b>Настройка бота!</b>\n\n'
    text += 'Чтобы пначать использовать бата и настроить его по Вашим предпочтениям ответьте на следующие вопросы.\n\n'
    text += '......................'

    keyboard = types.InlineKeyboardMarkup()
    key_begin = types.InlineKeyboardButton(text='🖊️ Начать', callback_data='begin')
    keyboard.add(key_begin)

    bot.send_message(message.chat.id, text=text, reply_markup=keyboard, parse_mode='HTML')

..............................

# Webhook
bot.set_webhook(url="https://myapp.pythonanywhere.com/" + TOKEN)

Для Flask (с ним не работал) есть такой пример:

import telebot
import os
from flask import Flask, request

bot = telebot.TeleBot(токен вашего бота)
server = Flask(__name__)


@bot.message_handler(commands=['start'])
def start(message):
    bot.reply_to(message, 'Hello, ' + message.from_user.first_name)

@bot.message_handler(func=lambda message: True, content_types=['text'])
def echo_message(message):
    bot.reply_to(message, message.text)


@server.route("/токен бота", methods=['POST'])
def getMessage():
    bot.process_new_updates([telebot.types.Update.de_json(request.stream.read().decode("utf-8"))])
    return "!", 200

@server.route("/")
def webhook():
    bot.remove_webhook()
    bot.set_webhook(url="https://ссылка на приложение/токен вашего бота")
    return "!", 200

server.run(host="0.0.0.0", port=os.environ.get('PORT', 5000))

Буду благодарен за помощь.

Рекомендуем хостинг TIMEWEB
Рекомендуем хостинг TIMEWEB
Стабильный хостинг, на котором располагается социальная сеть EVILEG. Для проектов на Django рекомендуем VDS хостинг.

Ол саған ұнайды ма? Әлеуметтік желілерде бөлісіңіз!

11
nayk1982
  • Там. 3, 2019, 8:25 Т.Ж.

Добрый день.
Посмотрите для начала статус установленного WebHook, нет ли ошибок с сертификатом безопасности. Это можно сделать из командной строки:

curl https://api.telegram.org/bot<TOKEN>/getWebhookInfo
    Arrow
    • Там. 4, 2019, 9:06 Т.Ж.

    Добрый день.

    Попробовал реализовать через сертефикат и изменил код:

    from telebot import TeleBot, types
    from os.path import join
    from django.conf import settings
    from rest_framework.response import Response
    from rest_framework.views import APIView
    from TelegramBot.settings import TOKEN
    
    
    bot = TeleBot(settings.TOKEN, threaded=False)
    
    
    class UpdateBot(APIView):
        def post(self, request):
            # Сюда должны получать сообщения от телеграм и далее обрабатываться ботом
            json_str = request.body.decode('UTF-8')
            update = types.Update.de_json(json_str)
            bot.process_new_updates([update])
    
            return Response({'code': 200})
    
    
    @bot.message_handler(commands=['start'])
    def start_message(message):
        # User написал /start в диалоге с ботом
        text = '<b>Настройка бота!</b>\n\n'
        text += 'Чтобы пначать использовать бата и настроить его по Вашим предпочтениям ответьте на следующие вопросы.\n\n'
        text += '......................'
    
        keyboard = types.InlineKeyboardMarkup()
        key_begin = types.InlineKeyboardButton(text='🖊️ Начать', callback_data='begin')
        keyboard.add(key_begin)
    
        bot.send_message(message.chat.id, text=text, reply_markup=keyboard, parse_mode='HTML')
    
    ..............................
    
    # Webhook
    CERTIFICATE = join(settings.BASE_DIR, 'PUBLIC.pem')
    bot.set_webhook(url="https://myapp.pythonanywhere.com/", certificate=open(CERTIFICATE),
                    max_connections=100000)
    

    Выполнил команду:

    curl https://api.telegram.org/bot<TOKEN>/getWebhookInfo
    

    Результат:

    {"ok":true,"result":{"url":"https://myapp.pythonanywhere.com/","has_custom_certificate":true,"pending_update_count":6,"last_error_date":1564923410,"last_error_message":"SSL error {error:1416F086:SSL routines:tls_process_server_certificate:certificate verify failed}","max_connections":100}}
    
      Arrow
      • Там. 4, 2019, 10:04 Т.Ж.

      В логах pythonenywhere:

      Request: method=get url=https://api.telegram.org/bot<KEY>/setWebhook params={'url': 'https://myapp.pythonanywhere.com/', 'max_connections': 100000} files={'certificate': <_io.TextIOWrapper name='/home/admin/Bot/PUBLIC.pem' mode='r' encoding='UTF-8'>}
      (apihelper.py:55 uWSGIWorker1Core0) DEBUG - TeleBot: "The server returned: 'b'{"ok":true,"result":true,"description":"Webhook was set"}''"
      The server returned: 'b'{"ok":true,"result":true,"description":"Webhook was set"}'
      

      Проверка:

      curl https://api.telegram.org/bot<KEY>/getWebhookInfo
      {"ok":true,"result":{"url":"https://myapp.pythonanywhere.com/","has_custom_certificate":true,"pending_update_count":7,"last_error_date":1564927364,"last_error_message":"SSL error {error:1416F086:SSL routines:tls_process_server_certificate:certificate verify failed}","max_connections":100}}
      
        nayk1982
        • Там. 4, 2019, 11:31 Т.Ж.

        Сертификат не проходит проверку. У меня такое было, когда пытался использовать самоподписанный сертификат. Решилось установкой сертификата Lets Encrypt. После этого WebHook заработал без проблем и сообщения стали приходить.

          Arrow
          • Там. 4, 2019, 1:07 Т.Қ.
          • (өңделген)

          Попробовал создать сертефикат при помощи acme-tiny (Let's Encrypt certificates):

          openssl genrsa 4096 > account.key
          

          В коде изменил:

          # Webhook
          CERTIFICATE = join(settings.BASE_DIR, 'account.key')
          bot.set_webhook(url="https://myapp.pythonanywhere.com/", certificate=open(CERTIFICATE),
                          max_connections=100000)
          

          И получил в логах:

          Request: method=get url=https://api.telegram.org/bot<KEY>/setWebhook params={'url': 'https://myapp.pythonanywhere.com/', 'max_connections': 100000} files={'certificate': <_io.TextIOWrapper name='/home/admin/Bot/account.key' mode='r' encoding='UTF-8'>}
          The server returned: 'b'{"ok":false,"error_code":400,"description":"Bad Request: bad webhook: Failed to set custom certificate file"}''
          

          Сервер его вообще не принимает. C рекомендуемым certbot то же самое.

            Arrow
            • Там. 5, 2019, 8:16 Т.Ж.

            Решил уйти от сертификата к простому токену.

            Изменил код на:

            Файл urls.py приложения бота:

            from django.urls import path
            from bot.views import UpdateBot
            from django.conf import settings
            from django.views.decorators.csrf import csrf_exempt
            
            
            urlpatterns = [
                path('{}'.format(settings.TOKEN), csrf_exempt(UpdateBot.as_view()), name='update'),
            ]
            
            

            Код views.py приложения бота:

            from rest_framework.response import Response
            from django.http import HttpResponse
            from django.views import View
            from telebot import TeleBot, types, logger
            from bot.models import User, Category, Content, Country, Link, User_category, User_content, SentLinks
            
            
            bot = TeleBot(settings.TOKEN, threaded=False)
            
            
            class UpdateBot(View):
                def get(self, request, *args, **kwargs):
                    return HttpResponse("Бот запусчен и работает.")
            
                def post(self, request, *args, **kwargs):
                    json_str = request.body.decode('UTF-8')
                    update = types.Update.de_json(json_str)
                    bot.process_new_updates([update])
            
                    return Response({'code': 200})
            
            
            @bot.message_handler(commands=['start'])
            def start_message(message):
                # User написал /start в диалоге с ботом
                text = '<b>Настройка бота!</b>\n\n'
                text += 'Чтобы пначать использовать бата и настроить его по Вашим предпочтениям ответьте на следующие вопросы.\n\n'
                text += '......................'
            
                keyboard = types.InlineKeyboardMarkup()
                key_begin = types.InlineKeyboardButton(text='🖊️ Начать', callback_data='begin')
                keyboard.add(key_begin)
            
                bot.send_message(message.chat.id, text=text, reply_markup=keyboard, parse_mode='HTML')
            
            ..............................
            
            # Webhook
            bot.remove_webhook()
            bot.set_webhook(url=f"{settings.DOMAIN}/{settings.TOKEN}")
            # DOMAIN = "https://myapp.pythonanywhere.com"
            # 
            

            Сразу, когда перезагрузил веб приложение в pythonenywhere пришло куча сообщений от бота (реакция на мои сообщения /start), а затем опять тишина.

            При проверке через терминал получаю:

            $ curl https://api.telegram.org/bot<KEY>/getWebhookInfo
            {"ok":true,"result":{"url":"https://myapp.pythonanywhere.com/<KEY>","has_custom_certificate":false,"pending_update_count":19,"last_error_date":1565006606,"last_err
            or_message":"Wrong response from the webhook: 500 Internal Server Error","max_connections":40}}
            

            В логах тишина. Немогу понять, что не верно. Особенно если по началу все вроде бы заработало.

              Arrow
              • Там. 5, 2019, 9:44 Т.Ж.

              Все заработало.

              Только теперь не знаю как быть. Толи я, что-то не то сделал, то ли телеграм тупит.
              Постоянно сробатывает Одна и таже процедура в коде, а именно часть 'Рассказать о боте':

              def menu(message):
                  try:
                      if message.text == 'Рассказать о боте':
                      """ Эта часть отрабатывает """
                          markup = types.InlineKeyboardMarkup()
                          forward_btn = types.InlineKeyboardButton(
                              text='...............',
                              url='https://t.me/share/url?url=https://t.me/' + settings.BOT_NAME)
                          markup.add(forward_btn)
                          text = '.............. '
                          bot.send_message(message.chat.id, text=text,
                                           reply_markup=markup, parse_mode='html')
                          bot.register_next_step_handler(message, menu)
              
                      elif message.text == 'Поиск':
                          bot.delete_message(message.chat.id, message.message_id)
                          markup = types.InlineKeyboardMarkup()
                          search_channel = types.InlineKeyboardButton(
                              text='.....', callback_data='.....')
                          search_bot = types.InlineKeyboardButton(
                              text='....', callback_data='.....')
                          search_chat = types.InlineKeyboardButton(
                              text='....', callback_data='....')
                          markup.row(search_channel)
                          markup.row(search_bot)
                          markup.row(search_chat)
                          text = '.......'
                          bot.send_message(message.chat.id, text=text,
                                           parse_mode='html', reply_markup=markup)
                          bot.register_next_step_handler(message, owner_menu)
              
                      elif message.text == 'Изменить настройки':
                          bot.register_next_step_handler(message, owner_menu)
                          manage_interests(message)
              
              
                      else:
                          raise Exception()
                  except:
                      bot.register_next_step_handler(message, owner_menu)
              

              Подскажите пожалуйста, где я туплю. Спасибо.

                nayk1982
                • Там. 5, 2019, 12:05 Т.Қ.

                Возможно телеграм повторяет одно и то же сообщение, это бывает, если в ответ на сообщение от телеграм ваш бот ничего не отвечает. Нужно на сообщения посылать ответ:

                {"ok": true}
                

                И еще, в каждом сообщении есть счетчик update_id, для каждого чата. Можно его запоминать и сравнивать, чтоб не дублировать ответы.

                  Arrow
                  • Там. 6, 2019, 7:07 Т.Ж.
                  • Жауап шешім ретінде белгіленді.

                  Спасибо.

                  Что-то не особо помогает.

                  Похоже что у pythonanywhere какие-то проблемы с передачей обновлений боту или у телеграм. То приходят данные, то нет. Сейчас проверка показывает:

                  {"ok":true,"result":{"url":"https://myapp.pythonanywhere.com/<Key>","has_custom_certificate":false,"pending_update_count":0,"max_connections":40}}
                  

                  А сообщения вообще перестали приходить.

                  Можете привести пример реализации ответа {"ok": true} или подредактировать мое, может я уже туплю на ровном месте.

                  Мой код:

                  urls.py приложения бота:

                  from django.urls import path
                  from django.conf import settings
                  from django.views.decorators.csrf import csrf_exempt
                  from bot.views import UpdateBot, Check
                  
                  app_name = 'PlayMarketBot'
                  
                  urlpatterns = [
                      path('{}'.format(settings.TOKEN), csrf_exempt(UpdateBot.as_view()), name='update'),
                      path('/', csrf_exempt(Check.as_view()), name='check'),
                  ]
                  

                  views.py приложения бота:

                  from django.conf import settings
                  from random import randint
                  from django.http import HttpResponse
                  from django.views import View
                  from telebot import TeleBot, types, logger
                  from bot.models import User, Category, Content, Country, Link, User_category, User_content, SentLinks
                  from threading import Timer
                  import logging
                  
                  logger.setLevel(logging.DEBUG)
                  
                  __author__ = '@masterarrow'
                  
                  bot = TeleBot(settings.TOKEN)
                  
                  update_id = None
                  
                  
                  class Check(View):
                      def get(self, request, *args, **kwargs):
                          return HttpResponse("Бот запусчен и работает.")
                  
                  
                  class UpdateBot(View):
                      def post(self, request, *args, **kwargs):
                          global update_id
                          json_str = request.body.decode('UTF-8')
                          update = types.Update.de_json(json_str)
                          if update_id != update.update_id:
                              bot.process_new_updates([update])
                              update_id = update.update_id
                  
                          return Response(b'{"ok":true,"result":[]}')
                  
                  
                  @bot.message_handler(commands=['start'])
                  def start_message(message):
                      bot.delete_message(message.chat.id, message.message_id)
                      text = '/////////////////////////'
                  
                      keyboard = types.InlineKeyboardMarkup()
                      key_begin = types.InlineKeyboardButton(text='🖊️ Начать', callback_data='begin')
                      keyboard.add(key_begin)
                  
                      bot.send_message(message.chat.id, text=text, reply_markup=keyboard, parse_mode='HTML')
                  
                  .............................................
                  
                  def menu(message):
                      try:
                          # Always working ???
                          if message.text == 'Рассказать':
                              markup = types.InlineKeyboardMarkup()
                              forward_btn = types.InlineKeyboardButton(
                                  text='Пригласить',
                                  url='https://t.me/share/url?url=https://t.me/' + settings.BOT_NAME)
                              markup.add(forward_btn)
                              text = '......................................'
                              bot.send_message(message.chat.id, text=text,
                                               reply_markup=markup, parse_mode='html')
                              bot.register_next_step_handler(message, menu)
                  
                          elif message.text == 'Поиск':
                              bot.delete_message(message.chat.id, message.message_id)
                              markup = types.InlineKeyboardMarkup()
                              search_channel = types.InlineKeyboardButton(
                                  text='......', callback_data='.............')
                              search_bot = types.InlineKeyboardButton(
                                  text='.......', callback_data='.........')
                              search_chat = types.InlineKeyboardButton(
                                  text='.....', callback_data='...........')
                              markup.row(search_channel)
                              markup.row(search_bot)
                              markup.row(search_chat)
                              text = '....................'
                              bot.send_message(message.chat.id, text=text,
                                               parse_mode='html', reply_markup=markup)
                              bot.register_next_step_handler(message, menu)
                  
                          elif message.text == 'Настроить':
                              manage_interests(message)
                              bot.register_next_step_handler(message, menu)
                  
                          else:
                              raise Exception()
                      except:
                          bot.register_next_step_handler(message, menu)
                  
                  
                  bot.remove_webhook()
                  bot.set_webhook(url=f"{settings.DOMAIN}/{settings.TOKEN}")
                  
                  

                  Большое спасибо за помощь.

                    nayk1982
                    • Там. 6, 2019, 10:04 Т.Қ.

                    На python не делал, поэтому не подскажу, как правильно.
                    Бросаются в глаза последние строчки из views.py:

                    bot.remove_webhook()
                    bot.set_webhook(url=f"{settings.DOMAIN}/{settings.TOKEN}")
                    

                    Они что, выполняются каждый раз? Если да, то это не правильно. WebHook устанавливается один раз и после этого ничего с ним делать не нужно (пока адрес вашего сервера не изменится или токен бота). Я бы вообще управление вэбхуком из кода убрал, это можно из консоли сделать одной строчкой, когда понадобится.

                    Могу рассказать, как делал я своего бота. На сервере для начала создал простой скрипт для приема сообщений и вывода в лог-файлы всего, что принял. Это поможет разобраться, как работает Telegram API. WebHook поставил на этот скрипт, добавил бота в тестовый чат Telegram и просто изучал, как работает API, параллельно читая документацию. Потом начал потихоньку наполнять этот скрипт и делать дополнительные если нужно. По началу лучше логировать все, что только можно, легче разобраться.

                      Arrow
                      • Там. 7, 2019, 2:24 Т.Ж.

                      Спасибо за подсказку.
                      Да я и не подумал, что код установки webhook можно вообще убрать или вынести.

                      Большое спасибо за помощь.

                        Пікірлер

                        Тек рұқсаты бар пайдаланушылар ғана пікір қалдыра алады.
                        Кіріңіз немесе Тіркеліңіз
                        AD

                        C++ - Тест 004. Указатели, Массивы и Циклы

                        • Нәтиже:50ұпай,
                        • Бағалау ұпайлары-4
                        m
                        • molni99
                        • Қаз. 26, 2024, 1:37 Т.Ж.

                        C++ - Тест 004. Указатели, Массивы и Циклы

                        • Нәтиже:80ұпай,
                        • Бағалау ұпайлары4
                        m
                        • molni99
                        • Қаз. 26, 2024, 1:29 Т.Ж.

                        C++ - Тест 004. Указатели, Массивы и Циклы

                        • Нәтиже:20ұпай,
                        • Бағалау ұпайлары-10
                        Соңғы пікірлер
                        ИМ
                        Игорь МаксимовҚар. 22, 2024, 11:51 Т.Ж.
                        Django - Оқулық 017. Теңшелген Django кіру беті Добрый вечер Евгений! Я сделал себе авторизацию аналогичную вашей, все работает, кроме возврата к предидущей странице. Редеректит всегда на главную, хотя в логах сервера вижу запросы на правильн…
                        Evgenii Legotckoi
                        Evgenii LegotckoiҚаз. 31, 2024, 2:37 Т.Қ.
                        Django - Сабақ 064. Python Markdown кеңейтімін қалай жазуға болады Добрый день. Да, можно. Либо через такие же плагины, либо с постобработкой через python библиотеку Beautiful Soup
                        A
                        ALO1ZEҚаз. 19, 2024, 8:19 Т.Ж.
                        Qt Creator көмегімен fb3 файл оқу құралы Подскажите как это запустить? Я не шарю в программировании и кодинге. Скачал и установаил Qt, но куча ошибок выдается и не запустить. А очень надо fb3 переконвертировать в html
                        ИМ
                        Игорь МаксимовҚаз. 5, 2024, 7:51 Т.Ж.
                        Django - Сабақ 064. Python Markdown кеңейтімін қалай жазуға болады Приветствую Евгений! У меня вопрос. Можно ли вставлять свои классы в разметку редактора markdown? Допустим имея стандартную разметку: <ul> <li></li> <li></l…
                        d
                        dblas5Шілде 5, 2024, 11:02 Т.Ж.
                        QML - Сабақ 016. SQLite деректер қоры және онымен QML Qt-та жұмыс істеу Здравствуйте, возникает такая проблема (я новичок): ApplicationWindow неизвестный элемент. (М300) для TextField и Button аналогично. Могу предположить, что из-за более новой верси…
                        Енді форумда талқылаңыз
                        m
                        moogoҚар. 22, 2024, 7:17 Т.Ж.
                        Mosquito Spray System Effective Mosquito Systems for Backyard | Eco-Friendly Misting Control Device & Repellent Spray - Moogo ; Upgrade your backyard with our mosquito-repellent device! Our misters conce…
                        Evgenii Legotckoi
                        Evgenii LegotckoiМаусым 24, 2024, 3:11 Т.Қ.
                        добавить qlineseries в функции Я тут. Работы оень много. Отправил его в бан.
                        t
                        tonypeachey1Қар. 15, 2024, 6:04 Т.Ж.
                        google domain [url=https://google.com/]domain[/url] domain [http://www.example.com link title]
                        NSProject
                        NSProjectМаусым 4, 2022, 3:49 Т.Ж.
                        Всё ещё разбираюсь с кешем. В следствии прочтения данной статьи. Я принял для себя решение сделать кеширование свойств менеджера модели LikeDislike. И так как установка evileg_core для меня не была возможна, ибо он писался…

                        Бізді әлеуметтік желілерде бақылаңыз