mirror of
https://github.com/civsocit/olgram.git
synced 2023-07-22 01:29:12 +03:00
Независимый проект Instance
This commit is contained in:
parent
60bb00bcc9
commit
c5e0192d24
10
bot.py
10
bot.py
@ -1,7 +1,6 @@
|
|||||||
import asyncio
|
import asyncio
|
||||||
|
|
||||||
import aiogram.types
|
import aiogram.types
|
||||||
import tortoise.transactions
|
|
||||||
from aiogram import Bot as AioBot, Dispatcher
|
from aiogram import Bot as AioBot, Dispatcher
|
||||||
from aiogram.contrib.fsm_storage.memory import MemoryStorage
|
from aiogram.contrib.fsm_storage.memory import MemoryStorage
|
||||||
|
|
||||||
@ -27,6 +26,14 @@ async def invite_callback(identify: int, message: aiogram.types.Message):
|
|||||||
await bot.group_chats.add(chat)
|
await bot.group_chats.add(chat)
|
||||||
|
|
||||||
|
|
||||||
|
async def left_callback(identify: int, message: aiogram.types.Message):
|
||||||
|
bot = await Bot.get(id=identify)
|
||||||
|
|
||||||
|
chat = await bot.group_chats.get_or_none(chat_id=message.chat.id)
|
||||||
|
if chat:
|
||||||
|
await bot.group_chats.remove(chat)
|
||||||
|
|
||||||
|
|
||||||
def run_bot(bot: BotInstance, loop: ty.Optional[asyncio.AbstractEventLoop] = None):
|
def run_bot(bot: BotInstance, loop: ty.Optional[asyncio.AbstractEventLoop] = None):
|
||||||
loop = loop or asyncio.get_event_loop()
|
loop = loop or asyncio.get_event_loop()
|
||||||
loop.create_task(bot.start_polling())
|
loop.create_task(bot.start_polling())
|
||||||
@ -39,6 +46,7 @@ async def run_all_bots(loop: asyncio.AbstractEventLoop):
|
|||||||
bot.super_chat_id,
|
bot.super_chat_id,
|
||||||
bot.start_text,
|
bot.start_text,
|
||||||
invite_callback=invite_callback,
|
invite_callback=invite_callback,
|
||||||
|
left_callback=left_callback,
|
||||||
identify=bot.id), loop)
|
identify=bot.id), loop)
|
||||||
|
|
||||||
|
|
||||||
|
@ -5,6 +5,6 @@ COPY . /app
|
|||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
|
|
||||||
RUN pip install --upgrade pip && \
|
RUN pip install --upgrade pip && \
|
||||||
pip install -r requirements.txt && \
|
pip install -r requirements.txt
|
||||||
|
|
||||||
CMD ["python", "instance"]
|
CMD ["python", "bot.py"]
|
||||||
|
@ -5,6 +5,8 @@ import aioredis
|
|||||||
from aiogram import Dispatcher, types, exceptions
|
from aiogram import Dispatcher, types, exceptions
|
||||||
from aiogram.contrib.fsm_storage.memory import MemoryStorage
|
from aiogram.contrib.fsm_storage.memory import MemoryStorage
|
||||||
|
|
||||||
|
from settings import InstanceSettings
|
||||||
|
|
||||||
|
|
||||||
class BotInstance:
|
class BotInstance:
|
||||||
def __init__(self, token: str, super_chat_id: int, start_text: str,
|
def __init__(self, token: str, super_chat_id: int, start_text: str,
|
||||||
@ -26,12 +28,22 @@ class BotInstance:
|
|||||||
self._dp.stop_polling()
|
self._dp.stop_polling()
|
||||||
|
|
||||||
async def start_polling(self):
|
async def start_polling(self):
|
||||||
self._redis = await aioredis.create_redis_pool('redis://localhost:6370')
|
self._redis = await aioredis.create_redis_pool(InstanceSettings.redis_path())
|
||||||
|
|
||||||
bot = aiogram.Bot(self._token)
|
bot = aiogram.Bot(self._token)
|
||||||
self._dp = Dispatcher(bot, storage=MemoryStorage())
|
self._dp = Dispatcher(bot, storage=MemoryStorage())
|
||||||
|
|
||||||
self._dp.register_message_handler(self._receive_text, content_types=[types.ContentType.TEXT])
|
# Здесь перечислены все типы сообщений, которые бот должен пересылать
|
||||||
|
self._dp.register_message_handler(self._receive_message, 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])
|
||||||
|
# Callback-и на добавление бота в чат и удаление бота из чата
|
||||||
self._dp.register_message_handler(self._receive_invite, content_types=[types.ContentType.NEW_CHAT_MEMBERS])
|
self._dp.register_message_handler(self._receive_invite, content_types=[types.ContentType.NEW_CHAT_MEMBERS])
|
||||||
self._dp.register_message_handler(self._receive_left, content_types=[types.ContentType.LEFT_CHAT_MEMBER])
|
self._dp.register_message_handler(self._receive_left, content_types=[types.ContentType.LEFT_CHAT_MEMBER])
|
||||||
|
|
||||||
@ -55,46 +67,50 @@ class BotInstance:
|
|||||||
if message.left_chat_member.id == message.bot.id:
|
if message.left_chat_member.id == message.bot.id:
|
||||||
await self._left_callback(self._identify, message)
|
await self._left_callback(self._identify, message)
|
||||||
|
|
||||||
async def _receive_text(self, message: types.Message):
|
async def _receive_message(self, message: types.Message):
|
||||||
"""
|
"""
|
||||||
Some text received
|
Получено обычное сообщение, вероятно, для пересыла в другой чат
|
||||||
:param message:
|
:param message:
|
||||||
:return:
|
:return:
|
||||||
"""
|
"""
|
||||||
if message.text and message.text.startswith("/start"):
|
if message.text and message.text.startswith("/start"):
|
||||||
|
# На команду start нужно ответить, не пересылая сообщение никуда
|
||||||
await message.answer(self._start_text)
|
await message.answer(self._start_text)
|
||||||
return
|
return
|
||||||
|
|
||||||
if message.chat.id != self._super_chat_id:
|
if message.chat.id != self._super_chat_id:
|
||||||
# Это обычный чат
|
# Это обычный чат: сообщение нужно переслать в супер-чат
|
||||||
new_message = await message.forward(self._super_chat_id)
|
new_message = await message.forward(self._super_chat_id)
|
||||||
await self._redis.set(self._message_unique_id(new_message.message_id), message.chat.id)
|
await self._redis.set(self._message_unique_id(new_message.message_id), message.chat.id)
|
||||||
else:
|
else:
|
||||||
# Это чат, в который бот должен пересылать сообщения
|
# Это супер-чат
|
||||||
if message.reply_to_message:
|
if message.reply_to_message:
|
||||||
|
# Ответ из супер-чата переслать тому пользователю,
|
||||||
chat_id = await self._redis.get(self._message_unique_id(message.reply_to_message.message_id))
|
chat_id = await self._redis.get(self._message_unique_id(message.reply_to_message.message_id))
|
||||||
if not chat_id:
|
if not chat_id:
|
||||||
chat_id = message.reply_to_message.forward_from_chat
|
chat_id = message.reply_to_message.forward_from_chat
|
||||||
if not chat_id:
|
if not chat_id:
|
||||||
await message.reply("Невозможно ответить, автор сообщения не найден")
|
await message.reply("Невозможно переслать сообщение: автор не найден")
|
||||||
return
|
return
|
||||||
chat_id = int(chat_id)
|
chat_id = int(chat_id)
|
||||||
try:
|
try:
|
||||||
await message.copy_to(chat_id)
|
await message.copy_to(chat_id)
|
||||||
except exceptions.MessageError:
|
except exceptions.MessageError:
|
||||||
await message.reply("Невозможно отправить сообщение пользователю: возможно, он заблокировал бота")
|
await message.reply("Невозможно переслать сообщение: возможно, автор заблокировал бота")
|
||||||
return
|
return
|
||||||
|
|
||||||
else:
|
else:
|
||||||
await message.forward(self._super_chat_id)
|
await message.forward(self._super_chat_id)
|
||||||
|
|
||||||
|
|
||||||
if __name__ == '__main__':
|
if __name__ == '__main__':
|
||||||
# Single instance mode
|
"""
|
||||||
import os
|
Режим single-instance. В этом режиме не работает olgram. На сервере запускается только один feedback (instance)
|
||||||
|
бот для пересылки сообщений. Все настройки этого бота задаются в переменных окружения на сервере. Бот работает
|
||||||
|
в режиме polling
|
||||||
|
"""
|
||||||
bot = BotInstance(
|
bot = BotInstance(
|
||||||
os.getenv("TOKEN"),
|
InstanceSettings.token(),
|
||||||
int(os.getenv("CHAT_ID")),
|
InstanceSettings.super_chat_id(),
|
||||||
os.getenv("START_TEXT")
|
InstanceSettings.start_text()
|
||||||
)
|
)
|
||||||
asyncio.get_event_loop().run_until_complete(bot.start_polling())
|
asyncio.get_event_loop().run_until_complete(bot.start_polling())
|
||||||
|
23
instance/docker-compose.yaml
Normal file
23
instance/docker-compose.yaml
Normal file
@ -0,0 +1,23 @@
|
|||||||
|
version: '3'
|
||||||
|
services:
|
||||||
|
redis:
|
||||||
|
restart: unless-stopped
|
||||||
|
image: 'bitnami/redis:latest'
|
||||||
|
environment:
|
||||||
|
- ALLOW_EMPTY_PASSWORD=yes
|
||||||
|
volumes:
|
||||||
|
- redis-db:/bitnami/redis/data
|
||||||
|
env_file:
|
||||||
|
- .env
|
||||||
|
instance:
|
||||||
|
build: .
|
||||||
|
restart: unless-stopped
|
||||||
|
depends_on:
|
||||||
|
- redis
|
||||||
|
env_file:
|
||||||
|
- .env
|
||||||
|
environment:
|
||||||
|
- INSTANCE_REDIS_PATH=redis://redis
|
||||||
|
|
||||||
|
volumes:
|
||||||
|
redis-db:
|
3
instance/requirements.txt
Normal file
3
instance/requirements.txt
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
aiogram
|
||||||
|
python-dotenv
|
||||||
|
aioredis
|
46
instance/settings.py
Normal file
46
instance/settings.py
Normal file
@ -0,0 +1,46 @@
|
|||||||
|
from dotenv import load_dotenv
|
||||||
|
import os
|
||||||
|
|
||||||
|
load_dotenv()
|
||||||
|
|
||||||
|
|
||||||
|
class InstanceSettings:
|
||||||
|
@classmethod
|
||||||
|
def _get_env(cls, parameter: str) -> str:
|
||||||
|
parameter = os.getenv(parameter, None)
|
||||||
|
if not parameter:
|
||||||
|
raise ValueError(f"{parameter} not defined in ENV")
|
||||||
|
return parameter
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def token(cls) -> str:
|
||||||
|
"""
|
||||||
|
Token instance бота
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
|
return cls._get_env("INSTANCE_TOKEN")
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def super_chat_id(cls) -> int:
|
||||||
|
"""
|
||||||
|
ID чата, в который бот пересылает сообщения
|
||||||
|
Это может быть личный чат (ID > 0) или общий чат (ID < 0)
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
|
return int(cls._get_env("INSTANCE_SUPER_CHAT_ID"))
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def start_text(cls) -> str:
|
||||||
|
"""
|
||||||
|
Этот текст будет отправляться пользователю по команде /start
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
|
return cls._get_env("INSTANCE_START_TEXT")
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def redis_path(cls) -> str:
|
||||||
|
"""
|
||||||
|
Путь до БД redis
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
|
return cls._get_env("INSTANCE_REDIS_PATH")
|
@ -1,12 +1,12 @@
|
|||||||
from abc import ABC
|
|
||||||
from dotenv import load_dotenv
|
from dotenv import load_dotenv
|
||||||
|
from abc import ABC
|
||||||
import os
|
import os
|
||||||
|
|
||||||
|
|
||||||
load_dotenv()
|
load_dotenv()
|
||||||
|
|
||||||
|
|
||||||
class _Settings(ABC):
|
class AbstractSettings(ABC):
|
||||||
@classmethod
|
@classmethod
|
||||||
def _get_env(cls, parameter: str) -> str:
|
def _get_env(cls, parameter: str) -> str:
|
||||||
parameter = os.getenv(parameter, None)
|
parameter = os.getenv(parameter, None)
|
||||||
@ -15,7 +15,7 @@ class _Settings(ABC):
|
|||||||
return parameter
|
return parameter
|
||||||
|
|
||||||
|
|
||||||
class OlgramSettings(_Settings):
|
class OlgramSettings(AbstractSettings):
|
||||||
@classmethod
|
@classmethod
|
||||||
def max_bots_per_user(cls) -> int:
|
def max_bots_per_user(cls) -> int:
|
||||||
"""
|
"""
|
||||||
@ -25,13 +25,17 @@ class OlgramSettings(_Settings):
|
|||||||
return 5
|
return 5
|
||||||
|
|
||||||
|
|
||||||
class BotSettings(_Settings):
|
class BotSettings(AbstractSettings):
|
||||||
@classmethod
|
@classmethod
|
||||||
def token(cls) -> str:
|
def token(cls) -> str:
|
||||||
|
"""
|
||||||
|
Токен olgram бота
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
return cls._get_env("BOT_TOKEN")
|
return cls._get_env("BOT_TOKEN")
|
||||||
|
|
||||||
|
|
||||||
class DatabaseSettings(_Settings):
|
class DatabaseSettings(AbstractSettings):
|
||||||
@classmethod
|
@classmethod
|
||||||
def user(cls) -> str:
|
def user(cls) -> str:
|
||||||
return cls._get_env("POSTGRES_USER")
|
return cls._get_env("POSTGRES_USER")
|
||||||
|
1
utils/settings.py
Normal file
1
utils/settings.py
Normal file
@ -0,0 +1 @@
|
|||||||
|
|
Loading…
Reference in New Issue
Block a user