mirror of
https://github.com/civsocit/olgram.git
synced 2023-07-22 01:29:12 +03:00
minor changes, auto-certificate
This commit is contained in:
parent
cf80e903da
commit
289b0d239a
5
.github/workflows/push.yaml
vendored
5
.github/workflows/push.yaml
vendored
@ -1,8 +1,5 @@
|
|||||||
name: Deploy
|
name: Deploy
|
||||||
on:
|
on: push
|
||||||
push:
|
|
||||||
branches:
|
|
||||||
- stable
|
|
||||||
env:
|
env:
|
||||||
IMAGE_NAME: bot
|
IMAGE_NAME: bot
|
||||||
USERNAME: mihalin
|
USERNAME: mihalin
|
||||||
|
45
docker-compose.yaml
Normal file
45
docker-compose.yaml
Normal file
@ -0,0 +1,45 @@
|
|||||||
|
version: '3'
|
||||||
|
services:
|
||||||
|
postgres:
|
||||||
|
image: postgres
|
||||||
|
restart: unless-stopped
|
||||||
|
env_file:
|
||||||
|
- .env
|
||||||
|
volumes:
|
||||||
|
- database:/var/lib/postgresql/data
|
||||||
|
networks:
|
||||||
|
- default
|
||||||
|
redis:
|
||||||
|
image: 'bitnami/redis:latest'
|
||||||
|
restart: unless-stopped
|
||||||
|
environment:
|
||||||
|
- ALLOW_EMPTY_PASSWORD=yes
|
||||||
|
volumes:
|
||||||
|
- redis-db:/bitnami/redis/data
|
||||||
|
env_file:
|
||||||
|
- .env
|
||||||
|
networks:
|
||||||
|
- default
|
||||||
|
olgram:
|
||||||
|
image: ghcr.io/civsocit/olgram/bot:stable
|
||||||
|
restart: unless-stopped
|
||||||
|
networks:
|
||||||
|
- default
|
||||||
|
env_file:
|
||||||
|
- .env
|
||||||
|
volumes:
|
||||||
|
- olgram-cert:/cert
|
||||||
|
ports:
|
||||||
|
- "${WEBHOOK_PORT}:80"
|
||||||
|
depends_on:
|
||||||
|
- postgres
|
||||||
|
- redis
|
||||||
|
|
||||||
|
volumes:
|
||||||
|
database:
|
||||||
|
redis-db:
|
||||||
|
olgram-cert:
|
||||||
|
|
||||||
|
networks:
|
||||||
|
default:
|
||||||
|
driver: bridge
|
@ -1,6 +1,12 @@
|
|||||||
#!/bin/sh
|
#!/bin/sh
|
||||||
set -e
|
set -e
|
||||||
|
|
||||||
|
if [[ -z "${CUSTOM_CERT}" ]]; then
|
||||||
|
if [ ! -f /cert/private.key ]; then
|
||||||
|
openssl req -newkey rsa:2048 -sha256 -nodes -keyout /cert/private.key -x509 -days 1000 -out /cert/public.pem -subj "/C=US/ST=Berlin/L=Berlin/O=my_org/CN=my_cn"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
sleep 10
|
sleep 10
|
||||||
aerich upgrade
|
aerich upgrade
|
||||||
python main.py
|
python main.py
|
12
example.env
Normal file
12
example.env
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
BOT_TOKEN=YOUR_BOT_TOKEN_HERE # example: 123456789:AAAA-abc123_AbcdEFghijKLMnopqrstu12
|
||||||
|
|
||||||
|
POSTGRES_USER=olgram
|
||||||
|
POSTGRES_PASSWORD=SOME_RANDOM_PASSWORD_HERE # example: x2y0n27ihiez93kmzj82
|
||||||
|
POSTGRES_DB=olgram
|
||||||
|
POSTGRES_HOST=postgres
|
||||||
|
|
||||||
|
WEBHOOK_HOST=YOUR_HOST_HERE # example: 11.143.142.140 or my_domain.com
|
||||||
|
WEBHOOK_PORT=8443 # allowed: 80, 443, 8080, 8443
|
||||||
|
CUSTOM_CERT=true # use that if you don't set up your own domain and let's encrypt certificate
|
||||||
|
|
||||||
|
REDIS_PATH=redis://redis
|
@ -1,53 +0,0 @@
|
|||||||
from aiogram import types
|
|
||||||
import asyncio
|
|
||||||
import aiocache
|
|
||||||
import typing as ty
|
|
||||||
from instance.bot import BotInstance, BotProperties
|
|
||||||
from olgram.models.models import Bot, GroupChat
|
|
||||||
|
|
||||||
|
|
||||||
class BotInstanceDatabase(BotInstance):
|
|
||||||
_instances: ty.Dict[int, "BotInstanceDatabase"] = dict()
|
|
||||||
|
|
||||||
def __init__(self, bot: Bot):
|
|
||||||
self._bot = bot
|
|
||||||
super().__init__()
|
|
||||||
|
|
||||||
@classmethod
|
|
||||||
async def run_all(cls):
|
|
||||||
bots = await Bot.all()
|
|
||||||
for bot in bots:
|
|
||||||
cls._instances[bot.id] = BotInstanceDatabase(bot)
|
|
||||||
# Polling только для отладки
|
|
||||||
asyncio.get_event_loop().create_task(cls._instances[bot.id].start_polling())
|
|
||||||
|
|
||||||
@aiocache.cached(ttl=5)
|
|
||||||
async def _properties(self) -> BotProperties:
|
|
||||||
await self._bot.refresh_from_db()
|
|
||||||
return BotProperties(self._bot.token, self._bot.start_text, int(self._bot.token.split(":")[0]),
|
|
||||||
await self._bot.super_chat_id())
|
|
||||||
|
|
||||||
async def _setup(self):
|
|
||||||
await super()._setup()
|
|
||||||
# Callback-и на добавление бота в чат и удаление бота из чата
|
|
||||||
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])
|
|
||||||
|
|
||||||
async def _receive_invite(self, message: types.Message):
|
|
||||||
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})
|
|
||||||
if chat not in await self._bot.group_chats.all():
|
|
||||||
await self._bot.group_chats.add(chat)
|
|
||||||
await self._bot.save()
|
|
||||||
break
|
|
||||||
|
|
||||||
async def _receive_left(self, message: types.Message):
|
|
||||||
if message.left_chat_member.id == message.bot.id:
|
|
||||||
chat = await self._bot.group_chats.filter(chat_id=message.chat.id).first()
|
|
||||||
if chat:
|
|
||||||
await self._bot.group_chats.remove(chat)
|
|
||||||
if self._bot.group_chat == chat:
|
|
||||||
self._bot.group_chat = None
|
|
||||||
await self._bot.save(update_fields=["group_chat"])
|
|
@ -1,10 +0,0 @@
|
|||||||
FROM python:3.8-buster
|
|
||||||
|
|
||||||
COPY . /app
|
|
||||||
|
|
||||||
WORKDIR /app
|
|
||||||
|
|
||||||
RUN pip install --upgrade pip && \
|
|
||||||
pip install -r requirements.txt
|
|
||||||
|
|
||||||
CMD ["python", "bot.py"]
|
|
123
instance/bot.py
123
instance/bot.py
@ -1,123 +0,0 @@
|
|||||||
import asyncio
|
|
||||||
import aiogram
|
|
||||||
import aioredis
|
|
||||||
from abc import ABC, abstractmethod
|
|
||||||
from dataclasses import dataclass
|
|
||||||
from aiogram import Dispatcher, types, exceptions
|
|
||||||
from aiogram.contrib.fsm_storage.memory import MemoryStorage
|
|
||||||
|
|
||||||
try:
|
|
||||||
from settings import InstanceSettings
|
|
||||||
except ModuleNotFoundError:
|
|
||||||
from .settings import InstanceSettings
|
|
||||||
|
|
||||||
|
|
||||||
@dataclass()
|
|
||||||
class BotProperties:
|
|
||||||
token: str
|
|
||||||
start_text: str
|
|
||||||
bot_id: int
|
|
||||||
super_chat_id: int
|
|
||||||
|
|
||||||
|
|
||||||
class BotInstance(ABC):
|
|
||||||
def __init__(self):
|
|
||||||
self._redis: aioredis.Redis = None
|
|
||||||
self._dp: aiogram.Dispatcher = None
|
|
||||||
|
|
||||||
@abstractmethod
|
|
||||||
async def _properties(self) -> BotProperties:
|
|
||||||
raise NotImplementedError()
|
|
||||||
|
|
||||||
def stop_polling(self):
|
|
||||||
print("stop polling")
|
|
||||||
self._dp.stop_polling()
|
|
||||||
|
|
||||||
async def _setup(self):
|
|
||||||
self._redis = await aioredis.create_redis_pool(InstanceSettings.redis_path())
|
|
||||||
|
|
||||||
props = await self._properties()
|
|
||||||
|
|
||||||
bot = aiogram.Bot(props.token)
|
|
||||||
self._dp = Dispatcher(bot, storage=MemoryStorage())
|
|
||||||
|
|
||||||
# Здесь перечислены все типы сообщений, которые бот должен пересылать
|
|
||||||
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])
|
|
||||||
|
|
||||||
async def start_polling(self):
|
|
||||||
print("start polling")
|
|
||||||
await self._setup()
|
|
||||||
await self._dp.start_polling()
|
|
||||||
|
|
||||||
@classmethod
|
|
||||||
def _message_unique_id(cls, bot_id: int, message_id: int) -> str:
|
|
||||||
return f"{bot_id}_{message_id}"
|
|
||||||
|
|
||||||
async def _receive_message(self, message: types.Message):
|
|
||||||
"""
|
|
||||||
Получено обычное сообщение, вероятно, для пересыла в другой чат
|
|
||||||
:param message:
|
|
||||||
:return:
|
|
||||||
"""
|
|
||||||
props = await self._properties()
|
|
||||||
if message.text and message.text.startswith("/start"):
|
|
||||||
# На команду start нужно ответить, не пересылая сообщение никуда
|
|
||||||
await message.answer(props.start_text)
|
|
||||||
return
|
|
||||||
|
|
||||||
if message.chat.id != props.super_chat_id:
|
|
||||||
# Это обычный чат: сообщение нужно переслать в супер-чат
|
|
||||||
new_message = await message.forward(props.super_chat_id)
|
|
||||||
await self._redis.set(self._message_unique_id(props.bot_id, new_message.message_id),
|
|
||||||
message.chat.id)
|
|
||||||
else:
|
|
||||||
# Это супер-чат
|
|
||||||
if message.reply_to_message:
|
|
||||||
# Ответ из супер-чата переслать тому пользователю,
|
|
||||||
chat_id = await self._redis.get(
|
|
||||||
self._message_unique_id(props.bot_id, message.reply_to_message.message_id))
|
|
||||||
if not chat_id:
|
|
||||||
chat_id = message.reply_to_message.forward_from_chat
|
|
||||||
if not chat_id:
|
|
||||||
await message.reply("Невозможно переслать сообщение: автор не найден")
|
|
||||||
return
|
|
||||||
chat_id = int(chat_id)
|
|
||||||
try:
|
|
||||||
await message.copy_to(chat_id)
|
|
||||||
except exceptions.MessageError:
|
|
||||||
await message.reply("Невозможно переслать сообщение: возможно, автор заблокировал бота")
|
|
||||||
return
|
|
||||||
else:
|
|
||||||
await message.forward(props.super_chat_id)
|
|
||||||
|
|
||||||
|
|
||||||
class FreezeBotInstance(BotInstance):
|
|
||||||
def __init__(self, token: str, start_text: str, super_chat_id: int):
|
|
||||||
super().__init__()
|
|
||||||
|
|
||||||
self._props = BotProperties(token, start_text, int(token.split(":")[0]), super_chat_id)
|
|
||||||
|
|
||||||
async def _properties(self) -> BotProperties:
|
|
||||||
return self._props
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == '__main__':
|
|
||||||
"""
|
|
||||||
Режим single-instance. В этом режиме не работает olgram. На сервере запускается только один feedback (instance)
|
|
||||||
бот для пересылки сообщений. Все настройки этого бота задаются в переменных окружения на сервере. Бот работает
|
|
||||||
в режиме polling
|
|
||||||
"""
|
|
||||||
bot = FreezeBotInstance(
|
|
||||||
InstanceSettings.token(),
|
|
||||||
InstanceSettings.start_text(),
|
|
||||||
InstanceSettings.super_chat_id()
|
|
||||||
)
|
|
||||||
asyncio.get_event_loop().run_until_complete(bot.start_polling())
|
|
@ -1,23 +0,0 @@
|
|||||||
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:
|
|
@ -1,3 +0,0 @@
|
|||||||
aiogram
|
|
||||||
python-dotenv
|
|
||||||
aioredis
|
|
@ -1,46 +0,0 @@
|
|||||||
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")
|
|
@ -8,9 +8,9 @@ load_dotenv()
|
|||||||
|
|
||||||
class AbstractSettings(ABC):
|
class AbstractSettings(ABC):
|
||||||
@classmethod
|
@classmethod
|
||||||
def _get_env(cls, parameter: str) -> str:
|
def _get_env(cls, parameter: str, allow_none: bool = False) -> str:
|
||||||
parameter = os.getenv(parameter, None)
|
parameter = os.getenv(parameter, None)
|
||||||
if not parameter:
|
if not parameter and not allow_none:
|
||||||
raise ValueError(f"{parameter} not defined in ENV")
|
raise ValueError(f"{parameter} not defined in ENV")
|
||||||
return parameter
|
return parameter
|
||||||
|
|
||||||
@ -54,6 +54,19 @@ class ServerSettings(AbstractSettings):
|
|||||||
"""
|
"""
|
||||||
return cls._get_env("REDIS_PATH")
|
return cls._get_env("REDIS_PATH")
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def use_custom_cert(cls) -> bool:
|
||||||
|
use = cls._get_env("CUSTOM_CERT", allow_none=True)
|
||||||
|
return use and "true" in use.lower()
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def priv_path(cls) -> str:
|
||||||
|
return "/cert/private.key"
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def public_path(cls) -> str:
|
||||||
|
return "/cert/public.pem"
|
||||||
|
|
||||||
|
|
||||||
class BotSettings(AbstractSettings):
|
class BotSettings(AbstractSettings):
|
||||||
@classmethod
|
@classmethod
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
aiogram
|
aiogram~=2.13
|
||||||
tortoise-orm[asyncpg]
|
tortoise-orm[asyncpg]
|
||||||
aerich==0.5.4
|
aerich==0.5.4
|
||||||
python-dotenv
|
python-dotenv~=0.17.1
|
||||||
aioredis==1.3.1
|
aioredis==1.3.1
|
||||||
aiocache
|
aiocache
|
||||||
|
aiohttp
|
@ -2,6 +2,7 @@ from aiogram import Bot as AioBot
|
|||||||
from olgram.models.models import Bot
|
from olgram.models.models import Bot
|
||||||
from aiohttp import web
|
from aiohttp import web
|
||||||
from asyncio import get_event_loop
|
from asyncio import get_event_loop
|
||||||
|
import ssl
|
||||||
from olgram.settings import ServerSettings
|
from olgram.settings import ServerSettings
|
||||||
from .custom import CustomRequestHandler
|
from .custom import CustomRequestHandler
|
||||||
|
|
||||||
@ -28,7 +29,10 @@ async def register_token(bot: Bot) -> bool:
|
|||||||
await unregister_token(bot.token)
|
await unregister_token(bot.token)
|
||||||
|
|
||||||
a_bot = AioBot(bot.token)
|
a_bot = AioBot(bot.token)
|
||||||
res = await a_bot.set_webhook(url_for_bot(bot))
|
certificate = None
|
||||||
|
if ServerSettings.use_custom_cert():
|
||||||
|
certificate = ServerSettings.public_path()
|
||||||
|
res = await a_bot.set_webhook(url_for_bot(bot), certificate=certificate)
|
||||||
await a_bot.session.close()
|
await a_bot.session.close()
|
||||||
del a_bot
|
del a_bot
|
||||||
return res
|
return res
|
||||||
@ -52,8 +56,13 @@ def main():
|
|||||||
app = web.Application()
|
app = web.Application()
|
||||||
app.router.add_route('*', r"/{name}", CustomRequestHandler, name='webhook_handler')
|
app.router.add_route('*', r"/{name}", CustomRequestHandler, name='webhook_handler')
|
||||||
|
|
||||||
|
context = None
|
||||||
|
if ServerSettings.use_custom_cert():
|
||||||
|
context = ssl.SSLContext(ssl.PROTOCOL_TLSv1_2)
|
||||||
|
context.load_cert_chain(ServerSettings.public_path(), ServerSettings.priv_path())
|
||||||
|
|
||||||
runner = web.AppRunner(app)
|
runner = web.AppRunner(app)
|
||||||
loop.run_until_complete(runner.setup())
|
loop.run_until_complete(runner.setup())
|
||||||
logger.info("Server initialization done")
|
logger.info("Server initialization done")
|
||||||
site = web.TCPSite(runner, host=ServerSettings.app_host(), port=ServerSettings.app_port())
|
site = web.TCPSite(runner, host=ServerSettings.app_host(), port=ServerSettings.app_port(), ssl_context=context)
|
||||||
return site
|
return site
|
||||||
|
Loading…
Reference in New Issue
Block a user