2021-09-10 21:32:06 +03:00
|
|
|
|
from aiogram import Bot as AioBot, Dispatcher
|
|
|
|
|
from aiogram.dispatcher.webhook import WebhookRequestHandler
|
|
|
|
|
from aiogram.dispatcher.webhook import SendMessage
|
2021-09-10 22:38:29 +03:00
|
|
|
|
from aiogram import exceptions
|
2021-09-10 21:32:06 +03:00
|
|
|
|
from aiogram import types
|
2021-09-10 22:07:02 +03:00
|
|
|
|
from contextvars import ContextVar
|
|
|
|
|
from aiohttp.web_exceptions import HTTPNotFound
|
2021-09-10 22:42:47 +03:00
|
|
|
|
from aioredis.commands import create_redis_pool
|
|
|
|
|
from aioredis import Redis
|
2022-02-18 21:47:40 +03:00
|
|
|
|
from tortoise.expressions import F
|
2021-09-17 11:07:55 +03:00
|
|
|
|
import logging
|
2021-09-10 22:38:29 +03:00
|
|
|
|
import typing as ty
|
|
|
|
|
from olgram.settings import ServerSettings
|
2022-01-18 23:28:03 +03:00
|
|
|
|
from olgram.models.models import Bot, GroupChat, BannedUser
|
2022-04-11 15:51:00 +03:00
|
|
|
|
from locales.locale import _, translators
|
2022-02-11 02:02:28 +03:00
|
|
|
|
from server.inlines import inline_handler
|
2021-09-17 11:35:51 +03:00
|
|
|
|
|
2021-09-17 11:07:55 +03:00
|
|
|
|
_logger = logging.getLogger(__name__)
|
2022-01-19 15:48:50 +03:00
|
|
|
|
_logger.setLevel(logging.INFO)
|
2021-09-17 11:07:55 +03:00
|
|
|
|
|
2021-09-10 22:07:02 +03:00
|
|
|
|
db_bot_instance: ContextVar[Bot] = ContextVar('db_bot_instance')
|
|
|
|
|
|
2021-09-10 22:42:47 +03:00
|
|
|
|
_redis: ty.Optional[Redis] = None
|
2021-09-10 22:38:29 +03:00
|
|
|
|
|
|
|
|
|
|
2022-04-11 15:51:00 +03:00
|
|
|
|
def _get_translator(message: types.Message) -> ty.Callable:
|
2022-04-11 17:24:13 +03:00
|
|
|
|
if not message.from_user.locale:
|
|
|
|
|
return _
|
2022-04-11 16:59:42 +03:00
|
|
|
|
return translators.get(message.from_user.locale.language, _)
|
2022-04-11 15:51:00 +03:00
|
|
|
|
|
|
|
|
|
|
2021-09-10 22:38:29 +03:00
|
|
|
|
async def init_redis():
|
|
|
|
|
global _redis
|
2021-09-10 22:42:47 +03:00
|
|
|
|
_redis = await create_redis_pool(ServerSettings.redis_path())
|
2021-09-10 22:38:29 +03:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def _message_unique_id(bot_id: int, message_id: int) -> str:
|
|
|
|
|
return f"{bot_id}_{message_id}"
|
|
|
|
|
|
2021-09-10 21:32:06 +03:00
|
|
|
|
|
2022-02-16 19:56:03 +03:00
|
|
|
|
def _thread_uniqie_id(bot_id: int, chat_id: int) -> str:
|
|
|
|
|
return f"thread_{bot_id}_{chat_id}"
|
|
|
|
|
|
|
|
|
|
|
2022-03-17 10:22:07 +03:00
|
|
|
|
def _on_security_policy(message: types.Message, bot):
|
2022-04-11 15:51:00 +03:00
|
|
|
|
_ = _get_translator(message)
|
2022-03-22 06:56:39 +03:00
|
|
|
|
text = _("<b>Политика конфиденциальности</b>\n\n"
|
|
|
|
|
"Этот бот не хранит ваши сообщения, имя пользователя и @username. При отправке сообщения (кроме команд "
|
|
|
|
|
"/start и /security_policy) ваш идентификатор пользователя записывается в кеш на некоторое время и потом "
|
|
|
|
|
"удаляется из кеша. Этот идентификатор используется только для общения с оператором; боты Olgram "
|
2022-03-22 05:43:10 +03:00
|
|
|
|
"не делают массовых рассылок.\n\n")
|
2022-03-17 09:44:24 +03:00
|
|
|
|
if bot.enable_additional_info:
|
2022-03-22 06:56:39 +03:00
|
|
|
|
text += _("При отправке сообщения (кроме команд /start и /security_policy) оператор <b>видит</b> ваши имя "
|
|
|
|
|
"пользователя, @username и идентификатор пользователя в силу настроек, которые оператор указал при "
|
2022-03-22 05:43:10 +03:00
|
|
|
|
"создании бота.")
|
2022-03-17 09:44:24 +03:00
|
|
|
|
else:
|
2022-03-22 06:56:39 +03:00
|
|
|
|
text += _("В зависимости от ваших настроек конфиденциальности Telegram, оператор может видеть ваш username, "
|
|
|
|
|
"имя пользователя и другую информацию.")
|
2022-03-17 09:44:24 +03:00
|
|
|
|
|
|
|
|
|
return SendMessage(chat_id=message.chat.id,
|
|
|
|
|
text=text,
|
|
|
|
|
parse_mode="HTML")
|
|
|
|
|
|
|
|
|
|
|
2022-03-17 08:26:14 +03:00
|
|
|
|
async def send_user_message(message: types.Message, super_chat_id: int, bot):
|
|
|
|
|
"""Переслать сообщение от пользователя, добавлять к нему user info при необходимости"""
|
|
|
|
|
if bot.enable_additional_info:
|
2022-03-22 05:43:10 +03:00
|
|
|
|
user_info = _("Сообщение от пользователя ")
|
2022-03-17 09:11:15 +03:00
|
|
|
|
user_info += message.from_user.full_name
|
2022-03-17 08:26:14 +03:00
|
|
|
|
if message.from_user.username:
|
|
|
|
|
user_info += " | @" + message.from_user.username
|
|
|
|
|
user_info += f" | #{message.from_user.id}"
|
2022-06-16 03:07:53 +03:00
|
|
|
|
|
|
|
|
|
# Добавлять информацию в конец текста
|
2022-06-16 03:22:15 +03:00
|
|
|
|
if message.content_type == types.ContentType.TEXT and len(message.text) + len(user_info) < 4093: # noqa:E721
|
2022-06-16 02:59:38 +03:00
|
|
|
|
new_message = await message.bot.send_message(super_chat_id, message.text + "\n\n" + user_info)
|
2022-06-16 03:07:53 +03:00
|
|
|
|
else: # Не добавлять информацию в конец текста, информация отдельным сообщением
|
2022-06-16 02:59:38 +03:00
|
|
|
|
new_message = await message.bot.send_message(super_chat_id, text=user_info)
|
|
|
|
|
new_message_2 = await message.copy_to(super_chat_id, reply_to_message_id=new_message.message_id)
|
|
|
|
|
await _redis.set(_message_unique_id(bot.pk, new_message_2.message_id), message.chat.id,
|
|
|
|
|
pexpire=ServerSettings.redis_timeout_ms())
|
2022-06-16 04:10:08 +03:00
|
|
|
|
await _redis.set(_message_unique_id(bot.pk, new_message.message_id), message.chat.id,
|
|
|
|
|
pexpire=ServerSettings.redis_timeout_ms())
|
2022-03-17 08:26:14 +03:00
|
|
|
|
return new_message
|
|
|
|
|
else:
|
2022-03-17 08:33:40 +03:00
|
|
|
|
new_message = await message.forward(super_chat_id)
|
|
|
|
|
await _redis.set(_message_unique_id(bot.pk, new_message.message_id), message.chat.id,
|
|
|
|
|
pexpire=ServerSettings.redis_timeout_ms())
|
|
|
|
|
return new_message
|
2021-09-10 22:38:29 +03:00
|
|
|
|
|
|
|
|
|
|
2022-03-17 08:26:14 +03:00
|
|
|
|
async def handle_user_message(message: types.Message, super_chat_id: int, bot):
|
|
|
|
|
"""Обычный пользователь прислал сообщение в бот, нужно переслать его операторам"""
|
2022-04-11 15:51:00 +03:00
|
|
|
|
_ = _get_translator(message)
|
2022-02-16 19:56:03 +03:00
|
|
|
|
is_super_group = super_chat_id < 0
|
2021-09-10 22:38:29 +03:00
|
|
|
|
|
2022-03-17 08:26:14 +03:00
|
|
|
|
# Проверить, не забанен ли пользователь
|
|
|
|
|
banned = await bot.banned_users.filter(telegram_id=message.chat.id)
|
|
|
|
|
if banned:
|
|
|
|
|
return SendMessage(chat_id=message.chat.id,
|
2022-03-22 05:43:10 +03:00
|
|
|
|
text=_("Вы заблокированы в этом боте"))
|
2022-01-18 23:28:03 +03:00
|
|
|
|
|
2022-03-17 08:26:14 +03:00
|
|
|
|
# Пересылаем сообщение в супер-чат
|
|
|
|
|
if is_super_group and bot.enable_threads:
|
|
|
|
|
thread_first_message = await _redis.get(_thread_uniqie_id(bot.pk, message.chat.id))
|
|
|
|
|
if thread_first_message:
|
|
|
|
|
# переслать в супер-чат, отвечая на предыдущее сообщение
|
|
|
|
|
try:
|
|
|
|
|
new_message = await message.copy_to(super_chat_id, reply_to_message_id=int(thread_first_message))
|
2022-03-18 00:21:44 +03:00
|
|
|
|
await _redis.set(_message_unique_id(bot.pk, new_message.message_id), message.chat.id,
|
|
|
|
|
pexpire=ServerSettings.redis_timeout_ms())
|
2022-03-17 08:26:14 +03:00
|
|
|
|
except exceptions.BadRequest:
|
|
|
|
|
new_message = await send_user_message(message, super_chat_id, bot)
|
2022-02-16 19:56:03 +03:00
|
|
|
|
await _redis.set(_thread_uniqie_id(bot.pk, message.chat.id), new_message.message_id,
|
|
|
|
|
pexpire=ServerSettings.thread_timeout_ms())
|
2022-03-17 08:26:14 +03:00
|
|
|
|
else:
|
|
|
|
|
# переслать супер-чат
|
|
|
|
|
new_message = await send_user_message(message, super_chat_id, bot)
|
|
|
|
|
await _redis.set(_thread_uniqie_id(bot.pk, message.chat.id), new_message.message_id,
|
|
|
|
|
pexpire=ServerSettings.thread_timeout_ms())
|
|
|
|
|
else: # личные сообщения не поддерживают потоки сообщений: простой forward
|
2022-03-18 00:21:44 +03:00
|
|
|
|
await send_user_message(message, super_chat_id, bot)
|
2022-02-16 19:56:03 +03:00
|
|
|
|
|
2022-03-17 08:26:14 +03:00
|
|
|
|
bot.incoming_messages_count = F("incoming_messages_count") + 1
|
|
|
|
|
await bot.save(update_fields=["incoming_messages_count"])
|
2022-03-17 08:05:03 +03:00
|
|
|
|
|
2022-03-17 08:26:14 +03:00
|
|
|
|
# И отправить пользователю специальный текст, если он указан
|
|
|
|
|
if bot.second_text:
|
|
|
|
|
return SendMessage(chat_id=message.chat.id, text=bot.second_text)
|
2022-02-19 05:38:03 +03:00
|
|
|
|
|
2022-03-17 08:26:14 +03:00
|
|
|
|
|
|
|
|
|
async def handle_operator_message(message: types.Message, super_chat_id: int, bot):
|
|
|
|
|
"""Оператор написал что-то, нужно переслать сообщение обратно пользователю, или забанить его и т.д."""
|
2022-04-11 15:51:00 +03:00
|
|
|
|
_ = _get_translator(message)
|
|
|
|
|
|
2022-03-17 08:26:14 +03:00
|
|
|
|
if message.reply_to_message:
|
2022-03-26 21:11:17 +03:00
|
|
|
|
|
|
|
|
|
if not message.reply_to_message.from_user.is_bot:
|
|
|
|
|
return # нас интересуют только ответы на сообщения бота
|
|
|
|
|
|
2022-03-17 08:26:14 +03:00
|
|
|
|
# В супер-чате кто-то ответил на сообщение пользователя, нужно переслать тому пользователю
|
|
|
|
|
chat_id = await _redis.get(_message_unique_id(bot.pk, message.reply_to_message.message_id))
|
|
|
|
|
if not chat_id:
|
|
|
|
|
chat_id = message.reply_to_message.forward_from_chat
|
|
|
|
|
if not chat_id:
|
|
|
|
|
return SendMessage(chat_id=message.chat.id,
|
2022-03-22 05:43:10 +03:00
|
|
|
|
text=_("<i>Невозможно переслать сообщение: автор не найден (сообщение слишком "
|
|
|
|
|
"старое?)</i>"),
|
2022-03-17 08:26:14 +03:00
|
|
|
|
parse_mode="HTML")
|
|
|
|
|
chat_id = int(chat_id)
|
|
|
|
|
|
|
|
|
|
if message.text == "/ban":
|
2022-03-22 05:43:10 +03:00
|
|
|
|
user, create = await BannedUser.get_or_create(telegram_id=chat_id, bot=bot)
|
2022-03-17 08:26:14 +03:00
|
|
|
|
await user.save()
|
2022-03-22 05:43:10 +03:00
|
|
|
|
return SendMessage(chat_id=message.chat.id, text=_("Пользователь заблокирован"))
|
2022-03-17 08:26:14 +03:00
|
|
|
|
|
|
|
|
|
if message.text == "/unban":
|
|
|
|
|
banned_user = await bot.banned_users.filter(telegram_id=chat_id).first()
|
|
|
|
|
if not banned_user:
|
2022-03-22 05:43:10 +03:00
|
|
|
|
return SendMessage(chat_id=message.chat.id, text=_("Пользователь не был забанен"))
|
2022-03-17 08:26:14 +03:00
|
|
|
|
else:
|
|
|
|
|
await banned_user.delete()
|
2022-03-22 05:43:10 +03:00
|
|
|
|
return SendMessage(chat_id=message.chat.id, text=_("Пользователь разбанен"))
|
2022-03-17 08:26:14 +03:00
|
|
|
|
|
|
|
|
|
try:
|
|
|
|
|
await message.copy_to(chat_id)
|
|
|
|
|
except (exceptions.MessageError, exceptions.Unauthorized):
|
2022-03-22 05:43:10 +03:00
|
|
|
|
await message.reply(_("<i>Невозможно переслать сообщение (автор заблокировал бота?)</i>"),
|
2022-03-17 08:26:14 +03:00
|
|
|
|
parse_mode="HTML")
|
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
bot.outgoing_messages_count = F("outgoing_messages_count") + 1
|
|
|
|
|
await bot.save(update_fields=["outgoing_messages_count"])
|
|
|
|
|
|
|
|
|
|
elif super_chat_id > 0:
|
|
|
|
|
# в супер-чате кто-то пишет сообщение сам себе, только для личных сообщений
|
|
|
|
|
await message.forward(super_chat_id)
|
2021-09-26 18:15:46 +03:00
|
|
|
|
# И отправить пользователю специальный текст, если он указан
|
|
|
|
|
if bot.second_text:
|
|
|
|
|
return SendMessage(chat_id=message.chat.id, text=bot.second_text)
|
2021-12-22 23:46:09 +03:00
|
|
|
|
|
2022-01-18 23:28:03 +03:00
|
|
|
|
|
2022-03-17 08:26:14 +03:00
|
|
|
|
async def message_handler(message: types.Message, *args, **kwargs):
|
|
|
|
|
_logger.info("message handler")
|
2022-04-11 15:51:00 +03:00
|
|
|
|
_ = _get_translator(message)
|
2022-03-17 08:26:14 +03:00
|
|
|
|
bot = db_bot_instance.get()
|
|
|
|
|
|
|
|
|
|
if message.text and message.text == "/start":
|
|
|
|
|
# На команду start нужно ответить, не пересылая сообщение никуда
|
2022-04-09 05:42:52 +03:00
|
|
|
|
text = bot.start_text
|
|
|
|
|
if bot.enable_olgram_text:
|
|
|
|
|
text += _(ServerSettings.append_text())
|
|
|
|
|
return SendMessage(chat_id=message.chat.id, text=text)
|
2022-03-17 08:26:14 +03:00
|
|
|
|
|
2022-03-17 09:44:24 +03:00
|
|
|
|
if message.text and message.text == "/security_policy":
|
|
|
|
|
# На команду security_policy нужно ответить, не пересылая сообщение никуда
|
|
|
|
|
return _on_security_policy(message, bot)
|
|
|
|
|
|
2022-03-17 08:26:14 +03:00
|
|
|
|
super_chat_id = await bot.super_chat_id()
|
|
|
|
|
|
|
|
|
|
if message.chat.id != super_chat_id:
|
|
|
|
|
# Это обычный чат
|
|
|
|
|
return await handle_user_message(message, super_chat_id, bot)
|
|
|
|
|
else:
|
|
|
|
|
# Это супер-чат
|
|
|
|
|
return await handle_operator_message(message, super_chat_id, bot)
|
2021-09-10 21:32:06 +03:00
|
|
|
|
|
|
|
|
|
|
2021-09-10 23:02:40 +03:00
|
|
|
|
async def receive_invite(message: types.Message):
|
|
|
|
|
bot = db_bot_instance.get()
|
|
|
|
|
for member in message.new_chat_members:
|
|
|
|
|
if member.id == message.bot.id:
|
|
|
|
|
chat, _ = await GroupChat.get_or_create(chat_id=message.chat.id,
|
|
|
|
|
defaults={"name": message.chat.full_name})
|
2021-09-17 11:35:51 +03:00
|
|
|
|
chat.name = message.chat.full_name
|
2021-09-17 11:49:48 +03:00
|
|
|
|
await chat.save()
|
2021-09-10 23:02:40 +03:00
|
|
|
|
if chat not in await bot.group_chats.all():
|
|
|
|
|
await bot.group_chats.add(chat)
|
|
|
|
|
await bot.save()
|
|
|
|
|
break
|
|
|
|
|
|
|
|
|
|
|
2021-12-15 00:14:49 +03:00
|
|
|
|
async def receive_group_create(message: types.Message):
|
|
|
|
|
bot = db_bot_instance.get()
|
|
|
|
|
|
|
|
|
|
chat, _ = await GroupChat.get_or_create(chat_id=message.chat.id,
|
|
|
|
|
defaults={"name": message.chat.full_name})
|
|
|
|
|
chat.name = message.chat.full_name
|
|
|
|
|
await chat.save()
|
|
|
|
|
if chat not in await bot.group_chats.all():
|
|
|
|
|
await bot.group_chats.add(chat)
|
|
|
|
|
await bot.save()
|
|
|
|
|
|
|
|
|
|
|
2021-09-10 23:02:40 +03:00
|
|
|
|
async def receive_left(message: types.Message):
|
|
|
|
|
bot = db_bot_instance.get()
|
|
|
|
|
if message.left_chat_member.id == message.bot.id:
|
|
|
|
|
chat = await bot.group_chats.filter(chat_id=message.chat.id).first()
|
|
|
|
|
if chat:
|
|
|
|
|
await bot.group_chats.remove(chat)
|
2021-09-17 11:16:18 +03:00
|
|
|
|
bot_group_chat = await bot.group_chat
|
|
|
|
|
if bot_group_chat == chat:
|
2021-09-10 23:02:40 +03:00
|
|
|
|
bot.group_chat = None
|
2021-09-17 11:35:51 +03:00
|
|
|
|
await bot.save()
|
2021-09-10 23:02:40 +03:00
|
|
|
|
|
|
|
|
|
|
2022-02-11 02:02:28 +03:00
|
|
|
|
async def receive_inline(inline_query):
|
2022-02-11 02:09:09 +03:00
|
|
|
|
_logger.info("inline handler")
|
2022-02-11 02:02:28 +03:00
|
|
|
|
bot = db_bot_instance.get()
|
|
|
|
|
return await inline_handler(inline_query, bot)
|
|
|
|
|
|
|
|
|
|
|
2021-12-14 23:55:19 +03:00
|
|
|
|
async def receive_migrate(message: types.Message):
|
|
|
|
|
bot = db_bot_instance.get()
|
|
|
|
|
from_id = message.chat.id
|
|
|
|
|
to_id = message.migrate_to_chat_id
|
|
|
|
|
|
|
|
|
|
chats = await bot.group_chats.filter(chat_id=from_id)
|
|
|
|
|
for chat in chats:
|
|
|
|
|
chat.chat_id = to_id
|
|
|
|
|
await chat.save(update_fields=["chat_id"])
|
|
|
|
|
|
|
|
|
|
|
2021-09-10 21:32:06 +03:00
|
|
|
|
class CustomRequestHandler(WebhookRequestHandler):
|
|
|
|
|
|
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
|
self._dispatcher = None
|
|
|
|
|
super(CustomRequestHandler, self).__init__(*args, **kwargs)
|
|
|
|
|
|
|
|
|
|
async def _create_dispatcher(self):
|
|
|
|
|
key = self.request.url.path[1:]
|
|
|
|
|
|
|
|
|
|
bot = await Bot.filter(code=key).first()
|
|
|
|
|
if not bot:
|
|
|
|
|
return None
|
2021-09-10 22:07:02 +03:00
|
|
|
|
db_bot_instance.set(bot)
|
2021-09-26 20:36:05 +03:00
|
|
|
|
dp = Dispatcher(AioBot(bot.decrypted_token()))
|
2021-09-10 21:32:06 +03:00
|
|
|
|
|
|
|
|
|
dp.register_message_handler(message_handler, content_types=[types.ContentType.TEXT,
|
|
|
|
|
types.ContentType.CONTACT,
|
|
|
|
|
types.ContentType.ANIMATION,
|
|
|
|
|
types.ContentType.AUDIO,
|
|
|
|
|
types.ContentType.DOCUMENT,
|
|
|
|
|
types.ContentType.PHOTO,
|
|
|
|
|
types.ContentType.STICKER,
|
|
|
|
|
types.ContentType.VIDEO,
|
|
|
|
|
types.ContentType.VOICE])
|
2021-09-10 23:02:40 +03:00
|
|
|
|
dp.register_message_handler(receive_invite, content_types=[types.ContentType.NEW_CHAT_MEMBERS])
|
|
|
|
|
dp.register_message_handler(receive_left, content_types=[types.ContentType.LEFT_CHAT_MEMBER])
|
2021-12-14 23:55:19 +03:00
|
|
|
|
dp.register_message_handler(receive_migrate, content_types=[types.ContentType.MIGRATE_TO_CHAT_ID])
|
2021-12-15 00:14:49 +03:00
|
|
|
|
dp.register_message_handler(receive_group_create, content_types=[types.ContentType.GROUP_CHAT_CREATED])
|
2022-02-11 02:02:28 +03:00
|
|
|
|
dp.register_inline_handler(receive_inline)
|
2021-09-10 21:32:06 +03:00
|
|
|
|
|
|
|
|
|
return dp
|
|
|
|
|
|
|
|
|
|
async def post(self):
|
2021-09-10 21:57:17 +03:00
|
|
|
|
dispatcher = await self._create_dispatcher()
|
2021-09-10 22:07:02 +03:00
|
|
|
|
if not dispatcher:
|
|
|
|
|
raise HTTPNotFound()
|
|
|
|
|
|
2021-09-10 21:57:17 +03:00
|
|
|
|
Dispatcher.set_current(dispatcher)
|
|
|
|
|
AioBot.set_current(dispatcher.bot)
|
|
|
|
|
return await super(CustomRequestHandler, self).post()
|
2021-09-10 21:32:06 +03:00
|
|
|
|
|
|
|
|
|
def get_dispatcher(self):
|
|
|
|
|
"""
|
|
|
|
|
Get Dispatcher instance from environment
|
|
|
|
|
|
|
|
|
|
:return: :class:`aiogram.Dispatcher`
|
|
|
|
|
"""
|
2021-09-10 21:57:17 +03:00
|
|
|
|
return Dispatcher.get_current()
|