123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691 |
- import re
- from io import BytesIO
- from sys import executable
- from struct import pack
- from asyncio import create_subprocess_shell
- from asyncio.subprocess import PIPE
- from datetime import datetime
- from ujson import dumps
- from tortoise.exceptions import IntegrityError
- from telethon.errors import MessageEmptyError
- from telethon.utils import get_display_name, get_peer_id
- from telethon.tl.types import MessageEntityCode
- from aiofiles.os import remove, path
- from aiohttp import ClientSession
- from emoji import is_emoji
- from cairosvg import svg2png
- from actions import (
- find_action,
- create_action,
- delete_action,
- add_gif,
- add_sticker,
- add_admin,
- delete_admin,
- add_or_update_birthday,
- get_birthdays,
- add_server,
- delete_server,
- add_allowed,
- delete_allowed,
- list_servers,
- get_server_ip,
- is_markov_enabled,
- enable_markov,
- disable_markov,
- set_markov_options,
- get_markov_option,
- markov_say,
- )
- from utils import (
- make_temporary_filename,
- make_cache_filename,
- parse_kind,
- get_user_name,
- calculate_age,
- unparse,
- remove_ansi_escapes,
- )
- class Handler:
- def __init__(self, handler, is_restricted=False, is_public=False):
- self.handler = handler
- self.is_restricted = is_restricted
- self.is_public = is_public
- async def newadmin_handler(bot, event, command):
- if command.argc < 1:
- await event.reply("Пожалуйста, укажите пользователя!")
- return
- try:
- target = await bot.get_entity(command.args[0])
- except ValueError:
- await event.reply("Недопустимое имя пользователя!")
- return
- try:
- await add_admin(target)
- except IntegrityError:
- await event.reply("Данный пользователь уже является администратором!")
- return
- await event.reply("Готово!~~")
- async def deladmin_handler(bot, event, command):
- if command.argc < 1:
- await event.reply("Пожалуйста, укажите пользователя!")
- return
- try:
- target = await bot.get_entity(command.args[0])
- except ValueError:
- await event.reply("Недопустимое имя пользователя!")
- return
- try:
- await delete_admin(target)
- except IndexError:
- await event.reply("Данный пользователь не является администратором!")
- return
- await event.reply("Готово!~~")
- async def newaction_handler(bot, event, command):
- if command.argc < 3:
- await event.reply("Пожалуйста, укажите тип, имя и шаблон действия!")
- return
- try:
- kind = parse_kind(command.args[0])
- except ValueError:
- await event.reply("Неверный тип действия!!!")
- return
- try:
- await create_action(command.args[1], " ".join(command.args[2:]), kind)
- except SyntaxError:
- await event.reply("Недопустимое имя действия!!!")
- return
- except IntegrityError:
- await event.reply("Действие с таким названием уже существует!")
- return
- await event.reply("Действие создано!")
- async def delaction_handler(bot, event, command):
- if command.argc < 1:
- await event.reply("Пожалуйста, укажите имя действия!")
- return
- try:
- await delete_action(command.args[0])
- except SyntaxError:
- await event.reply("Недопустимое имя действия!!!")
- return
- except NameError:
- await event.reply("Действия с таким названием не существует!")
- return
- await event.reply("Действие удалено!")
- async def addgif_handler(bot, event, command):
- if command.argc < 1:
- await event.reply("Пожалуйста, укажите имя действия!")
- return
- gif = await event.get_reply_message()
- if not gif or not gif.gif:
- await event.reply("Пожалуйста, добавьте GIF!")
- return
- try:
- action = await find_action(command.args[0])
- await add_gif(action, gif.file.id)
- except SyntaxError:
- await event.reply("Недопустимое имя действия!!!")
- return
- except NameError:
- await event.reply("Нет такого действия!")
- return
- await event.reply("Готово!~~")
- async def addserver_handler(bot, event, command):
- if command.argc < 2:
- await event.reply("Пожалуйста, укажите имя и адрес сервера!")
- return
- try:
- await add_server(command.args[0], command.args[1])
- except SyntaxError:
- await event.reply("Недопустимое имя сервера!!")
- return
- except ValueError:
- await event.reply("Пожалуйста, введите корректный IPv4-/IPv6-адрес!")
- return
- except IntegrityError:
- await event.reply("Данный сервер уже был добавлен ранее!")
- return
- await event.reply("Готово!~~")
- async def delserver_handler(bot, event, command):
- if command.argc < 1:
- await event.reply("Пожалуйста, укажите имя сервера!")
- return
- try:
- await delete_server(command.args[0])
- except SyntaxError:
- await event.reply("Недопустимое имя сервера!!")
- return
- except IndexError:
- await event.reply("Сервер с таким именем не найден!")
- return
- await event.reply("Готово!~~")
- async def allow_handler(bot, event, command):
- try:
- await add_allowed(get_peer_id(event.peer_id))
- except IntegrityError:
- await event.reply("Данный чат уже добавлен в список разрешённых!")
- return
- await event.reply("Готово!~~")
- async def disallow_handler(bot, event, command):
- try:
- await delete_allowed(get_peer_id(event.peer_id))
- except IndexError:
- await event.reply("Данный чат не найден в списке разрешённых!!")
- return
- await event.reply("Готово!~~")
- async def markov_handler(bot, event, command):
- if command.argc < 1:
- await event.reply("Некорректный синтаксис команды!!")
- return
- peer_id = get_peer_id(event.peer_id)
- if command.args[0] == "enable":
- try:
- await enable_markov(peer_id)
- except:
- await event.reply("Ошибка!!!!!")
- return
- await event.reply("Готово!~~")
- elif command.args[0] == "disable":
- try:
- await disable_markov(peer_id)
- except:
- await event.reply("Ошибка!!!!!")
- return
- await event.reply("Готово!~~")
- elif command.args[0] == "is_enabled":
- await event.reply(str(await is_markov_enabled(peer_id)))
- elif command.args[0] == "set" and command.argc == 3:
- try:
- await set_markov_options(
- peer_id, **{command.args[1]: float(command.args[2])}
- )
- except:
- await event.reply("Ошибка!!!!!")
- return
- await event.reply("Готово!~~")
- elif command.args[0] == "get" and command.argc == 2:
- try:
- await event.reply(str(await get_markov_option(peer_id, command.args[1])))
- except:
- await event.reply("Ошибка!!!!!")
- elif command.args[0] == "say":
- if not bot.markov.is_ready:
- await event.reply("Not ready:(")
- else:
- await markov_say(bot, peer_id)
- elif command.args[0] == "reply":
- if not bot.markov.is_ready:
- await event.reply("Not ready:(")
- else:
- await markov_say(bot, peer_id, reply_to=event)
- elif command.args[0] == "is_ready":
- await event.reply(str(bot.markov.is_ready))
- elif command.args[0] == "corpus_size":
- await event.reply(str(len(bot.markov.corpus)))
- elif command.args[0] == "rebuild":
- bot.markov.rebuild()
- await event.reply("Готово!~~")
- elif command.args[0] == "counter":
- await event.reply(str(bot.markov.counter))
- elif command.args[0] == "counter_reset":
- bot.markov.counter = 0
- await event.reply("Готово!~~")
- elif command.args[0] == "reset_corpus":
- bot.markov.corpus = []
- await event.reply("Готово!~~")
- elif command.args[0] == "unready":
- bot.markov.chain = None
- await event.reply("Готово!~~")
- elif command.args[0] == "save":
- bot.markov.save()
- await event.reply("Готово!~~")
- else:
- await event.reply("Некорректный синтаксис команды!!!")
- # Very, very, VERY evil code...
- async def make_message_shot(bot, message):
- if message.sender is None:
- sender_id = message.peer_id.channel_id
- full_name = await bot.get_entity(sender_id)
- full_name = full_name.title
- else:
- sender_id = message.sender.id
- full_name = get_display_name(message.sender)
- output_path = make_temporary_filename("png")
- avatar_path = await make_cache_filename(sender_id, "png")
- if not await path.isfile(avatar_path):
- await bot.download_profile_photo(sender_id, file=avatar_path, download_big=True)
- # TO-DO: make it better.
- mproc = await create_subprocess_shell(f"mogrify -format png {avatar_path}")
- await mproc.communicate()
- if not await path.isfile(avatar_path):
- avatar_path = "./resources/placeholder.png"
- data = bytes()
- data += pack("I", len(output_path))
- data += bytes(output_path, encoding="ASCII")
- data += pack("I", len(avatar_path))
- data += bytes(avatar_path, encoding="ASCII")
- username = bytes(full_name, encoding="UTF-8")
- data += pack("I", len(username))
- data += username
- data += pack("I", sender_id % 7)
- text = bytes(
- unparse(
- message.raw_text, [entity for entity, _ in message.get_entities_text()]
- ),
- encoding="UTF-8",
- )
- data += pack("I", len(text))
- data += text
- proc = await create_subprocess_shell("./makeshot/makeshot", stdin=PIPE)
- await proc.communicate(input=data)
- pproc = await create_subprocess_shell(f"pngcrush -reduce -ow {output_path}")
- await pproc.communicate()
- return output_path
- async def save_handler(bot, event, command):
- message = await event.get_reply_message()
- if not message:
- await event.reply("Пожалуйста, укажите сообщение для сохранения!")
- return
- emoji = "⚡"
- if command.argc >= 1:
- emoji = command.args[0]
- if len(emoji) not in range(1, 6) or not all(map(is_emoji, emoji)):
- await event.reply("Указан некорректный эмодзи!!!")
- return
- path = await make_message_shot(bot, message)
- try:
- file = await add_sticker(bot, path, emoji)
- await bot.send_file(message.peer_id, file=file, reply_to=message)
- finally:
- await remove(path)
- async def bday_handler(bot, event, command):
- if command.argc >= 1:
- try:
- date = datetime.strptime(" ".join(command.args), "%d.%m.%Y")
- except ValueError:
- await event.reply(
- "Дата не может быть распознана. Пожалуйста, введите свой день рождения в следующем формате: 01.01.1970 (день, месяц, год)."
- )
- return
- if date >= datetime.now():
- await event.reply("День рождения не может быть в будущем...")
- return
- if await add_or_update_birthday(get_peer_id(event.peer_id), event.sender, date):
- await event.reply("День рождения успешно добавлен!!!")
- else:
- await event.reply("День рождения успешно обновлён!!!")
- return
- birthdays = await get_birthdays(get_peer_id(event.peer_id))
- if not birthdays:
- await event.reply("Пока пусто...")
- return
- birthdays = map(
- lambda birthday: (birthday.user_id, calculate_age(birthday.date)), birthdays
- )
- birthdays = sorted(birthdays, key=lambda birthday: birthday[1].days_until)
- birthdays_list = ""
- for user_id, age in birthdays:
- try:
- entity = await bot.get_entity(user_id)
- except ValueError:
- await bot.get_participants(await event.get_chat())
- try:
- entity = await bot.get_entity(user_id)
- except ValueError:
- continue
- birthdays_list += f"{get_user_name(entity)} ❯ {age.age_now} ➔ {age.age} ❯ {age.date_string} ❯ {age.days_until}\n"
- await event.reply(f"Дни рождения:\n\n{birthdays_list}")
- async def vpn_handler(bot, event, command):
- if command.argc < 1:
- await event.reply(
- f"Пожалуйста, укажите имя сервера! Доступные сервера: {await list_servers()}."
- )
- return
- try:
- ip = await get_server_ip(command.args[0])
- except SyntaxError:
- await event.reply("Недопустимое имя сервера!!")
- return
- except IndexError:
- await event.reply("Сервер с таким именем не найден!")
- return
- if event.sender is None:
- sender_id = event.peer_id.channel_id
- else:
- sender_id = event.sender.id
- async with ClientSession() as session:
- try:
- async with session.post(
- f"http://{ip}:9217/api/obtain", data={"user_id": sender_id}
- ) as resp:
- data = await resp.json()
- profile = data["profile"]
- except:
- await event.reply(
- "Произошла ошибка при попытке обращения к API сервера… :("
- )
- return
- try:
- await bot.send_message(
- await bot.get_entity(sender_id),
- f"Ваш файл конфигурации WireGuard (сервер {command.args[0]}):\n\n```{profile}```",
- )
- except:
- await event.reply(
- "Произошла ошибка при отправке файла конфигурации в Ваши личные сообщения… :с"
- )
- return
- await event.reply(
- "Готово!!~~ Файл конфигурации WireGuard отправлен в Ваши личные сообщения!"
- )
- async def run_handler(bot, event, command):
- if command.argc < 1:
- async with ClientSession() as session:
- try:
- async with session.get("https://farlands.txlyre.website/langs") as resp:
- text = await resp.read()
- text = text.decode("UTF-8")
- await event.reply(f"Доступные языки:\n`{text}`")
- except:
- await event.reply("Произошла ошибка при попытке обращения к API… :(")
- return
- match = re.match(r"^(\w+)(?:\s|\n)((?:\n|.)*)$", command.args_string)
- if not match:
- await event.reply("Пожалуйста, не оставляйте ввод пустым!")
- return
- language_name, text = match.groups()
- if text.startswith("```") and text.endswith("```"):
- text = text[3:-3]
- text = text.replace("\xa0", " ") # i hate telegram
- async with ClientSession() as session:
- try:
- async with session.post(
- f"https://farlands.txlyre.website/run/{language_name}", data=text
- ) as resp:
- if resp.status in (404, 500):
- info = await resp.json()
- await event.reply(
- f'Произошла ошибка при попытке обращения к API… :(\nОтвет API: {info["detail"]}'
- )
- return
- elif resp.status != 200:
- await event.reply(
- "Сервер API временно недоступен. Пожалуйста, попробуйте ещё раз чуть позже."
- )
- return
- text = await resp.read()
- text = text.decode("UTF-8")[:4096]
- except:
- await event.reply("Произошла ошибка при попытке обращения к API… :(")
- return
- text = remove_ansi_escapes(text).strip()
- text = filter(
- lambda c: (c in " \t\n" or ord(c) >= 32) and ord(c) not in range(128, 159), text
- )
- text = "".join(text)
- text = text.replace("`", "")
- if not text:
- await event.reply("<пусто>")
- return
- try:
- try:
- await event.reply(f"```\n{text}```")
- except ValueError:
- await event.reply(text, parse_mode=None)
- except MessageEmptyError:
- await event.reply("<Telegram не смог декодировать текст сообщения>")
- async def sylvy_handler(bot, event, command):
- if command.argc < 1:
- await event.reply("Пожалуйста, не оставляйте ввод пустым!")
- return
- async with ClientSession() as session:
- try:
- async with session.post(
- "https://sylvy-engine.txlyre.website/api/compute",
- json={"program": command.args_string, "stringify": True},
- ) as resp:
- data = await resp.json()
- if data["status"] != "ok":
- await event.reply(f'Ошибка API Sylvy: {data["data"]["message"]}')
- return
- result = data["data"]["result"]
- except:
- await event.reply("Произошла ошибка при попытке обращения к API Sylvy… :(")
- return
- if result["status"] == "TIMEOUT":
- await event.reply(
- f"Максимальное время исполнения истекло (более тридцати секунд)!!!"
- )
- return
- elif result["status"] != "SUCCESS":
- await event.reply(f'Ошибка исполнения!!!\n\n```{result["stdout"]}```')
- return
- if result["plots"]:
- plots = []
- for plot in result["plots"]:
- buffer = BytesIO()
- svg2png(bytestring=plot, write_to=buffer)
- plots.append(buffer.getvalue())
- await bot.send_file(event.peer_id, file=plots, reply_to=event)
- return
- text = ""
- if result["stdout"]:
- text += result["stdout"]
- text += "\n"
- text += result["output"]
- text = text.rstrip()
- await event.reply(
- text, formatting_entities=[MessageEntityCode(offset=0, length=len(text))]
- )
- async def say_handler(bot, event, command):
- if not bot.markov.is_ready:
- await event.reply("Генератор текста ещё не готов к использованию. Пожалуйста, попробуйте чуть позже.")
- else:
- init_state = None
- if command.argc > 0:
- init_state = command.args_string
- try:
- await markov_say(bot, get_peer_id(event.peer_id), init_state=init_state)
- except:
- await event.reply("Ошибка :(")
- COMMANDS = {
- "newadmin": Handler(newadmin_handler, is_restricted=True),
- "deladmin": Handler(deladmin_handler, is_restricted=True),
- "newaction": Handler(newaction_handler, is_restricted=True),
- "delaction": Handler(delaction_handler, is_restricted=True),
- "addgif": Handler(addgif_handler, is_restricted=True),
- "addserver": Handler(addserver_handler, is_restricted=True),
- "delserver": Handler(delserver_handler, is_restricted=True),
- "allow": Handler(allow_handler, is_restricted=True),
- "disallow": Handler(disallow_handler, is_restricted=True),
- "markov": Handler(markov_handler, is_restricted=True),
- "save": Handler(save_handler),
- "bday": Handler(bday_handler),
- "vpn": Handler(vpn_handler),
- "sylvy": Handler(sylvy_handler, is_public=True),
- "run": Handler(run_handler, is_public=True),
- "say": Handler(say_handler, is_public=True),
- }
|