mirror of
https://github.com/civsocit/olgram.git
synced 2025-12-17 07:46:17 +00:00
Миграции
This commit is contained in:
@@ -1,90 +0,0 @@
|
||||
from aiogram import types, Bot as AioBot
|
||||
from aiogram.dispatcher import FSMContext
|
||||
from aiogram.utils.callback_data import CallbackData
|
||||
from textwrap import dedent
|
||||
|
||||
from olgram.utils.router import Router
|
||||
from olgram.utils.mix import try_delete_message
|
||||
from olgram.models.models import Bot, User
|
||||
|
||||
router = Router()
|
||||
|
||||
# Пользователь выбрал бота
|
||||
select_bot = CallbackData('bot_select', 'bot_id')
|
||||
# Пользователь выбрал, что хочет сделать со своим ботом
|
||||
bot_operation = CallbackData('bot_operation', 'bot_id', 'operation')
|
||||
# Пользователь выбрал чат
|
||||
select_bot_chat = CallbackData('chat_select', 'bot_id', 'chat_id')
|
||||
|
||||
|
||||
@router.callback_query_handler(select_bot.filter(), state="*")
|
||||
async def select_bot_callback(call: types.CallbackQuery, callback_data: dict, state: FSMContext):
|
||||
"""
|
||||
Пользователь выбрал бота для редактирования
|
||||
"""
|
||||
bot_id = callback_data["bot_id"]
|
||||
bot = await Bot.get_or_none(id=bot_id)
|
||||
if not bot or (await bot.owner).telegram_id != call.from_user.id:
|
||||
await call.answer("Такого бота нет, либо он принадлежит не вам", show_alert=True)
|
||||
return
|
||||
|
||||
await try_delete_message(call.message)
|
||||
|
||||
keyboard = types.InlineKeyboardMarkup(row_width=2)
|
||||
keyboard.insert(types.InlineKeyboardButton(text="Текст",
|
||||
callback_data=bot_operation.new(bot_id=bot_id, operation="text")))
|
||||
keyboard.insert(types.InlineKeyboardButton(text="Чат",
|
||||
callback_data=bot_operation.new(bot_id=bot_id, operation="chat")))
|
||||
keyboard.insert(types.InlineKeyboardButton(text="Удалить бот",
|
||||
callback_data=bot_operation.new(bot_id=bot_id, operation="delete")))
|
||||
keyboard.insert(types.InlineKeyboardButton(text="<<Вернуться к списку ботов",
|
||||
callback_data=bot_operation.new(bot_id=bot_id, operation="back")))
|
||||
|
||||
await AioBot.get_current().send_message(call.message.chat.id, dedent(f"""
|
||||
Управление ботом @{bot.name}.
|
||||
|
||||
Если у вас возникли вопросы по настройке бота, то посмотрите нашу справку /help.
|
||||
"""), reply_markup=keyboard)
|
||||
|
||||
|
||||
@router.callback_query_handler(bot_operation.filter(operation="delete"), state="*")
|
||||
async def delete_bot_callback(call: types.CallbackQuery, callback_data: dict, state: FSMContext):
|
||||
bot_id = callback_data["bot_id"]
|
||||
bot = await Bot.get_or_none(id=bot_id)
|
||||
if not bot or (await bot.owner).telegram_id != call.from_user.id:
|
||||
await call.answer("Такого бота нет, либо он принадлежит не вам", show_alert=True)
|
||||
return
|
||||
|
||||
await bot.delete()
|
||||
await call.answer("Бот удалён")
|
||||
await try_delete_message(call.message)
|
||||
|
||||
|
||||
@router.callback_query_handler(bot_operation.filter(operation="chat"), state="*")
|
||||
async def chats_bot_callback(call: types.CallbackQuery, callback_data: dict, state: FSMContext):
|
||||
bot_id = callback_data["bot_id"]
|
||||
bot = await Bot.get_or_none(id=bot_id)
|
||||
if not bot or (await bot.owner).telegram_id != call.from_user.id:
|
||||
await call.answer("Такого бота нет, либо он принадлежит не вам", show_alert=True)
|
||||
return
|
||||
|
||||
await try_delete_message(call.message)
|
||||
|
||||
keyboard = types.InlineKeyboardMarkup(row_width=2)
|
||||
|
||||
chats = await bot.group_chats.all()
|
||||
|
||||
if not chats:
|
||||
return await AioBot.get_current().send_message(call.message.chat.id, dedent(f"""
|
||||
Этот бот не добавлен в чаты, поэтому все сообщения будут приходить вам в бот.
|
||||
Чтобы подключить чат — просто добавьте бот @{bot.name} в чат.
|
||||
"""), reply_markup=keyboard)
|
||||
|
||||
for chat in chats:
|
||||
keyboard.insert(types.InlineKeyboardButton(text=chat.name,
|
||||
callback_data=select_bot_chat.new(bot_id=bot_id, chat_id=chat.id)))
|
||||
|
||||
await AioBot.get_current().send_message(call.message.chat.id, dedent(f"""
|
||||
В этом разделе вы можете привязать бота @{bot.name} к чату.
|
||||
Выберите чат, куда бот будет пересылать сообщения.
|
||||
"""), reply_markup=keyboard)
|
||||
142
olgram/commands/bot.py
Normal file
142
olgram/commands/bot.py
Normal file
@@ -0,0 +1,142 @@
|
||||
"""
|
||||
Здесь работа с конкретным ботом
|
||||
"""
|
||||
from aiogram import types, Bot as AioBot
|
||||
from aiogram.dispatcher import FSMContext
|
||||
from aiogram.utils.callback_data import CallbackData
|
||||
from textwrap import dedent
|
||||
|
||||
from olgram.utils.router import Router
|
||||
from olgram.utils.mix import try_delete_message
|
||||
from olgram.models.models import Bot, User
|
||||
|
||||
router = Router()
|
||||
|
||||
# Пользователь выбрал бота
|
||||
select_bot = CallbackData('bot_select', 'bot_id')
|
||||
# Пользователь выбрал, что хочет сделать со своим ботом
|
||||
bot_operation = CallbackData('bot_operation', 'bot_id', 'operation')
|
||||
# Пользователь выбрал чат
|
||||
select_bot_chat = CallbackData('chat_select', 'bot_id', 'chat_id')
|
||||
# Пользователь выбрал чат - личные сообщения
|
||||
select_bot_chat_personal = CallbackData('chat_select_personal', 'bot_id')
|
||||
|
||||
|
||||
def check_bot_owner(handler):
|
||||
"""
|
||||
Этот декоратор запрещает пользователям вызывать callback's (inline кнопки) для ботов, которыми они не владеют
|
||||
"""
|
||||
async def wrapped(call: types.CallbackQuery, callback_data: dict, state: FSMContext):
|
||||
bot_id = callback_data["bot_id"]
|
||||
bot = await Bot.get_or_none(id=bot_id)
|
||||
if not bot or (await bot.owner).telegram_id != call.from_user.id:
|
||||
await call.answer("У вас нет прав на этого бота", show_alert=True)
|
||||
return
|
||||
|
||||
return await handler(bot, call, callback_data, state)
|
||||
return wrapped
|
||||
|
||||
|
||||
@router.callback_query_handler(select_bot.filter(), state="*")
|
||||
@check_bot_owner
|
||||
async def select_bot_callback(bot: Bot, call: types.CallbackQuery, callback_data: dict, state: FSMContext):
|
||||
"""
|
||||
Пользователь выбрал бота для редактирования
|
||||
"""
|
||||
await try_delete_message(call.message)
|
||||
|
||||
keyboard = types.InlineKeyboardMarkup(row_width=2)
|
||||
keyboard.insert(types.InlineKeyboardButton(text="Текст",
|
||||
callback_data=bot_operation.new(bot_id=bot.id, operation="text")))
|
||||
keyboard.insert(types.InlineKeyboardButton(text="Чат",
|
||||
callback_data=bot_operation.new(bot_id=bot.id, operation="chat")))
|
||||
keyboard.insert(types.InlineKeyboardButton(text="Удалить бот",
|
||||
callback_data=bot_operation.new(bot_id=bot.id, operation="delete")))
|
||||
keyboard.insert(types.InlineKeyboardButton(text="<<Вернуться к списку ботов",
|
||||
callback_data=bot_operation.new(bot_id=bot.id, operation="back")))
|
||||
|
||||
await AioBot.get_current().send_message(call.message.chat.id, dedent(f"""
|
||||
Управление ботом @{bot.name}.
|
||||
|
||||
Если у вас возникли вопросы по настройке бота, то посмотрите нашу справку /help.
|
||||
"""), reply_markup=keyboard)
|
||||
|
||||
|
||||
@router.callback_query_handler(bot_operation.filter(operation="delete"), state="*")
|
||||
@check_bot_owner
|
||||
async def delete_bot_callback(bot: Bot, call: types.CallbackQuery, callback_data: dict, state: FSMContext):
|
||||
"""
|
||||
Кнопка "удалить" для бота
|
||||
"""
|
||||
await bot.delete()
|
||||
await call.answer("Бот удалён")
|
||||
await try_delete_message(call.message)
|
||||
|
||||
|
||||
@router.callback_query_handler(bot_operation.filter(operation="chat"), state="*")
|
||||
@check_bot_owner
|
||||
async def chats_bot_callback(bot: Bot, call: types.CallbackQuery, callback_data: dict, state: FSMContext):
|
||||
"""
|
||||
Кнопка "чаты" для бота
|
||||
"""
|
||||
await try_delete_message(call.message)
|
||||
|
||||
keyboard = types.InlineKeyboardMarkup(row_width=2)
|
||||
|
||||
chats = await bot.group_chats.all()
|
||||
|
||||
if not chats:
|
||||
return await AioBot.get_current().send_message(call.message.chat.id, dedent(f"""
|
||||
Этот бот не добавлен в чаты, поэтому все сообщения будут приходить вам в бот.
|
||||
Чтобы подключить чат — просто добавьте бот @{bot.name} в чат.
|
||||
"""), reply_markup=keyboard)
|
||||
|
||||
for chat in chats:
|
||||
keyboard.insert(types.InlineKeyboardButton(text=chat.name,
|
||||
callback_data=select_bot_chat.new(bot_id=bot.id, chat_id=chat.id)))
|
||||
keyboard.insert(types.InlineKeyboardButton(text="Личные сообщения",
|
||||
callback_data=select_bot_chat_personal.new(bot_id=bot.id)))
|
||||
await AioBot.get_current().send_message(call.message.chat.id, dedent(f"""
|
||||
В этом разделе вы можете привязать бота @{bot.name} к чату.
|
||||
Выберите чат, куда бот будет пересылать сообщения.
|
||||
"""), reply_markup=keyboard)
|
||||
|
||||
|
||||
@router.callback_query_handler(select_bot_chat.filter(), state="*")
|
||||
@check_bot_owner
|
||||
async def chat_selected_callback(bot: Bot, call: types.CallbackQuery, callback_data: dict, state: FSMContext):
|
||||
"""
|
||||
Пользователь выбрал групповой чат для бота
|
||||
"""
|
||||
chat_id = callback_data["chat_id"]
|
||||
chat = await bot.group_chats.filter(id=chat_id).first()
|
||||
if not chat:
|
||||
await call.answer("Нельзя привязать бота к этому чату")
|
||||
return
|
||||
bot.group_chat = chat
|
||||
await bot.save()
|
||||
await call.answer(f"Выбран чат {chat.name}")
|
||||
|
||||
|
||||
@router.callback_query_handler(select_bot_chat_personal.filter(), state="*")
|
||||
@check_bot_owner
|
||||
async def chat_selected_personal_callback(bot: Bot, call: types.CallbackQuery, callback_data: dict, state: FSMContext):
|
||||
"""
|
||||
Пользователь выбрал личный чат для бота
|
||||
"""
|
||||
bot.group_chat = None
|
||||
await bot.save()
|
||||
await call.answer(f"Выбран личный чат")
|
||||
|
||||
|
||||
@router.callback_query_handler(bot_operation.filter(operation="text"), state="*")
|
||||
@check_bot_owner
|
||||
async def text_bot_callback(bot: Bot, call: types.CallbackQuery, callback_data: dict, state: FSMContext):
|
||||
"""
|
||||
Кнопка "текст" для бота
|
||||
"""
|
||||
await AioBot.get_current().send_message(call.message.chat.id, dedent(f"""
|
||||
Текущий текст бота по кнопке start:
|
||||
|
||||
{bot.start_text}
|
||||
"""))
|
||||
@@ -1,3 +1,6 @@
|
||||
"""
|
||||
Здесь работа с ботами на первом уровне вложенности: список ботов, добавление ботов
|
||||
"""
|
||||
from aiogram import types, Bot as AioBot
|
||||
from aiogram.dispatcher import FSMContext
|
||||
from aiogram.utils.exceptions import Unauthorized, TelegramAPIError
|
||||
@@ -1,3 +1,7 @@
|
||||
"""
|
||||
Здесь простые команды на первом уровне вложенности: /start /help
|
||||
"""
|
||||
|
||||
from aiogram import types
|
||||
from aiogram.dispatcher import FSMContext
|
||||
from textwrap import dedent
|
||||
30
olgram/migrations/models/0_20210711121349_init.sql
Normal file
30
olgram/migrations/models/0_20210711121349_init.sql
Normal file
@@ -0,0 +1,30 @@
|
||||
-- upgrade --
|
||||
CREATE TABLE IF NOT EXISTS "group_chat" (
|
||||
"id" SERIAL NOT NULL PRIMARY KEY,
|
||||
"chat_id" INT NOT NULL UNIQUE,
|
||||
"name" VARCHAR(50) NOT NULL
|
||||
);
|
||||
CREATE INDEX IF NOT EXISTS "idx_group_chat_chat_id_5da32d" ON "group_chat" ("chat_id");
|
||||
CREATE TABLE IF NOT EXISTS "user" (
|
||||
"id" SERIAL NOT NULL PRIMARY KEY,
|
||||
"telegram_id" INT NOT NULL UNIQUE
|
||||
);
|
||||
CREATE INDEX IF NOT EXISTS "idx_user_telegra_66ffbd" ON "user" ("telegram_id");
|
||||
CREATE TABLE IF NOT EXISTS "bot" (
|
||||
"id" SERIAL NOT NULL PRIMARY KEY,
|
||||
"token" VARCHAR(50) NOT NULL UNIQUE,
|
||||
"name" VARCHAR(33) NOT NULL,
|
||||
"start_text" TEXT NOT NULL,
|
||||
"group_chat_id" INT REFERENCES "group_chat" ("id") ON DELETE SET NULL,
|
||||
"owner_id" INT NOT NULL REFERENCES "user" ("id") ON DELETE CASCADE
|
||||
);
|
||||
CREATE TABLE IF NOT EXISTS "aerich" (
|
||||
"id" SERIAL NOT NULL PRIMARY KEY,
|
||||
"version" VARCHAR(255) NOT NULL,
|
||||
"app" VARCHAR(20) NOT NULL,
|
||||
"content" JSONB NOT NULL
|
||||
);
|
||||
CREATE TABLE IF NOT EXISTS "bot_group_chat" (
|
||||
"bot_id" INT NOT NULL REFERENCES "bot" ("id") ON DELETE SET NULL,
|
||||
"groupchat_id" INT NOT NULL REFERENCES "group_chat" ("id") ON DELETE SET NULL
|
||||
);
|
||||
@@ -11,11 +11,19 @@ class Bot(Model):
|
||||
name = fields.CharField(max_length=33)
|
||||
start_text = fields.TextField(default=dedent("""
|
||||
Здравствуйте!
|
||||
Напишите ваш вопрос и мы ответим Вам в ближайшее время.
|
||||
Напишите ваш вопрос и мы ответим вам в ближайшее время.
|
||||
"""))
|
||||
|
||||
super_chat_id = fields.IntField()
|
||||
group_chats = fields.ManyToManyField("models.GroupChat", related_name="bots", on_delete=fields.relational.SET_NULL)
|
||||
group_chat = fields.ForeignKeyField("models.GroupChat", related_name="active_bots",
|
||||
on_delete=fields.relational.SET_NULL,
|
||||
null=True)
|
||||
|
||||
async def super_chat_id(self):
|
||||
group_chat = await self.group_chat
|
||||
if group_chat:
|
||||
return group_chat.chat_id
|
||||
return (await self.owner).telegram_id
|
||||
|
||||
class Meta:
|
||||
table = 'bot'
|
||||
|
||||
@@ -47,3 +47,19 @@ class DatabaseSettings(AbstractSettings):
|
||||
@classmethod
|
||||
def database_name(cls) -> str:
|
||||
return cls._get_env("POSTGRES_DB")
|
||||
|
||||
@classmethod
|
||||
def host(cls) -> str:
|
||||
return cls._get_env("POSTGRES_HOST")
|
||||
|
||||
|
||||
TORTOISE_ORM = {
|
||||
"connections": {"default": f'postgres://{DatabaseSettings.user()}:{DatabaseSettings.password()}'
|
||||
f'@{DatabaseSettings.host()}/{DatabaseSettings.database_name()}'},
|
||||
"apps": {
|
||||
"models": {
|
||||
"models": ["olgram.models.models", "aerich.models"],
|
||||
"default_connection": "default",
|
||||
},
|
||||
},
|
||||
}
|
||||
@@ -1,15 +0,0 @@
|
||||
from tortoise import Tortoise
|
||||
from olgram.settings import DatabaseSettings
|
||||
|
||||
|
||||
async def init_database():
|
||||
# Here we create a SQLite DB using file "db.sqlite3"
|
||||
# also specify the app name of "models"
|
||||
# which contain models from "app.models"
|
||||
await Tortoise.init(
|
||||
db_url=f'postgres://{DatabaseSettings.user()}:{DatabaseSettings.password()}'
|
||||
f'@localhost:5430/{DatabaseSettings.database_name()}',
|
||||
modules={'models': ['olgram.models.models']}
|
||||
)
|
||||
# Generate the schema
|
||||
await Tortoise.generate_schemas()
|
||||
Reference in New Issue
Block a user