commands.py 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491
  1. from io import BytesIO
  2. from sys import executable
  3. from struct import pack
  4. from asyncio import create_subprocess_shell
  5. from asyncio.subprocess import PIPE
  6. from datetime import datetime
  7. from ujson import dumps
  8. from tortoise.exceptions import IntegrityError
  9. from telethon.utils import get_display_name, get_peer_id
  10. from aiofiles.os import (
  11. remove,
  12. path
  13. )
  14. from aiohttp import ClientSession
  15. from emoji import is_emoji
  16. from cairosvg import svg2png
  17. from actions import (
  18. find_action,
  19. create_action,
  20. delete_action,
  21. add_gif,
  22. add_sticker,
  23. add_admin,
  24. delete_admin,
  25. add_or_update_birthday,
  26. get_birthdays,
  27. add_server,
  28. delete_server,
  29. add_allowed,
  30. delete_allowed,
  31. list_servers,
  32. get_server_ip
  33. )
  34. from utils import (
  35. WORDS_TABLE,
  36. make_temporary_filename,
  37. make_cache_filename,
  38. parse_kind,
  39. get_user_name,
  40. get_word_for,
  41. calculate_age,
  42. unparse
  43. )
  44. class Handler:
  45. def __init__(self, handler, is_restricted=False):
  46. self.handler = handler
  47. self.is_restricted = is_restricted
  48. async def newadmin_handler(bot, event, command):
  49. if command.argc < 1:
  50. await event.reply('Пожалуйста, укажите пользователя!')
  51. return
  52. try:
  53. target = await bot.get_entity(command.args[0])
  54. except ValueError:
  55. await event.reply('Недопустимое имя пользователя!')
  56. return
  57. try:
  58. await add_admin(target)
  59. except IntegrityError:
  60. await event.reply('Данный пользователь уже является администратором!')
  61. return
  62. await event.reply('Готово!~~')
  63. async def deladmin_handler(bot, event, command):
  64. if command.argc < 1:
  65. await event.reply('Пожалуйста, укажите пользователя!')
  66. return
  67. try:
  68. target = await bot.get_entity(command.args[0])
  69. except ValueError:
  70. await event.reply('Недопустимое имя пользователя!')
  71. return
  72. try:
  73. await delete_admin(target)
  74. except IndexError:
  75. await event.reply('Данный пользователь не является администратором!')
  76. return
  77. await event.reply('Готово!~~')
  78. async def newaction_handler(bot, event, command):
  79. if command.argc < 3:
  80. await event.reply('Пожалуйста, укажите тип, имя и шаблон действия!')
  81. return
  82. try:
  83. kind = parse_kind(command.args[0])
  84. except ValueError:
  85. await event.reply('Неверный тип действия!!!')
  86. return
  87. try:
  88. await create_action(command.args[1], ' '.join(command.args[2:]), kind)
  89. except SyntaxError:
  90. await event.reply('Недопустимое имя действия!!!')
  91. return
  92. except IntegrityError:
  93. await event.reply('Действие с таким названием уже существует!')
  94. return
  95. await event.reply('Действие создано!')
  96. async def delaction_handler(bot, event, command):
  97. if command.argc < 1:
  98. await event.reply('Пожалуйста, укажите имя действия!')
  99. return
  100. try:
  101. await delete_action(command.args[0])
  102. except SyntaxError:
  103. await event.reply('Недопустимое имя действия!!!')
  104. return
  105. except NameError:
  106. await event.reply('Действия с таким названием не существует!')
  107. return
  108. await event.reply('Действие удалено!')
  109. async def addgif_handler(bot, event, command):
  110. if command.argc < 1:
  111. await event.reply('Пожалуйста, укажите имя действия!')
  112. return
  113. gif = await event.get_reply_message()
  114. if not gif or not gif.gif:
  115. await event.reply('Пожалуйста, добавьте GIF!')
  116. return
  117. try:
  118. action = await find_action(command.args[0])
  119. await add_gif(action, gif.file.id)
  120. except SyntaxError:
  121. await event.reply('Недопустимое имя действия!!!')
  122. return
  123. except NameError:
  124. await event.reply('Нет такого действия!')
  125. return
  126. await event.reply('Готово!~~')
  127. async def addserver_handler(bot, event, command):
  128. if command.argc < 2:
  129. await event.reply('Пожалуйста, укажите имя и адрес сервера!')
  130. return
  131. try:
  132. await add_server(command.args[0], command.args[1])
  133. except SyntaxError:
  134. await event.reply('Недопустимое имя сервера!!')
  135. return
  136. except ValueError:
  137. await event.reply('Пожалуйста, введите корректный IPv4-/IPv6-адрес!')
  138. return
  139. except IntegrityError:
  140. await event.reply('Данный сервер уже был добавлен ранее!')
  141. return
  142. await event.reply('Готово!~~')
  143. async def delserver_handler(bot, event, command):
  144. if command.argc < 1:
  145. await event.reply('Пожалуйста, укажите имя сервера!')
  146. return
  147. try:
  148. await delete_server(command.args[0])
  149. except SyntaxError:
  150. await event.reply('Недопустимое имя сервера!!')
  151. return
  152. except IndexError:
  153. await event.reply('Сервер с таким именем не найден!')
  154. return
  155. await event.reply('Готово!~~')
  156. async def allow_handler(bot, event, command):
  157. try:
  158. await add_allowed(get_peer_id(event.peer_id))
  159. except IntegrityError:
  160. await event.reply('Данный чат уже добавлен в список разрешённых!')
  161. return
  162. await event.reply('Готово!~~')
  163. async def disallow_handler(bot, event, command):
  164. try:
  165. await delete_allowed(get_peer_id(event.peer_id))
  166. except IndexError:
  167. await event.reply('Данный чат не найден в списке разрешённых!!')
  168. return
  169. await event.reply('Готово!~~')
  170. # Very, very, VERY evil code...
  171. async def make_message_shot(bot, message):
  172. if message.sender is None:
  173. sender_id = message.peer_id.channel_id
  174. full_name = await bot.get_entity(sender_id)
  175. full_name = full_name.title
  176. else:
  177. sender_id = message.sender.id
  178. full_name = get_display_name(message.sender)
  179. output_path = make_temporary_filename('png')
  180. avatar_path = await make_cache_filename(sender_id, 'png')
  181. if not await path.isfile(avatar_path):
  182. await bot.download_profile_photo(sender_id, file=avatar_path)
  183. # TO-DO: make it better.
  184. mproc = await create_subprocess_shell(
  185. f'mogrify -format png {avatar_path}'
  186. )
  187. await mproc.communicate()
  188. data = bytes()
  189. data += pack('I', len(output_path))
  190. data += bytes(output_path, encoding='ASCII')
  191. data += pack('I', len(avatar_path))
  192. data += bytes(avatar_path, encoding='ASCII')
  193. username = bytes(full_name, encoding='UTF-8')
  194. data += pack('I', len(username))
  195. data += username
  196. data += pack('I', sender_id % 7)
  197. text = bytes(unparse(message.raw_text, [entity for entity, _ in message.get_entities_text()]), encoding='UTF-8')
  198. data += pack('I', len(text))
  199. data += text
  200. proc = await create_subprocess_shell(
  201. './makeshot/makeshot',
  202. stdin=PIPE
  203. )
  204. await proc.communicate(input=data)
  205. pproc = await create_subprocess_shell(
  206. f'pngcrush -reduce -ow {output_path} {output_path}'
  207. )
  208. await pproc.communicate()
  209. return output_path
  210. async def save_handler(bot, event, command):
  211. message = await event.get_reply_message()
  212. if not message:
  213. await event.reply('Пожалуйста, укажите сообщение для сохранения!')
  214. return
  215. emoji = '⚡'
  216. if command.argc >= 1:
  217. emoji = command.args[0]
  218. if len(emoji) not in range(1, 6)\
  219. or not all(map(is_emoji, emoji)):
  220. await event.reply('Указан некорректный эмодзи!!!')
  221. return
  222. path = await make_message_shot(bot, message)
  223. try:
  224. file = await add_sticker(bot, path, emoji)
  225. await bot.send_file(
  226. message.peer_id,
  227. file=file,
  228. reply_to=message
  229. )
  230. finally:
  231. await remove(path)
  232. async def bday_handler(bot, event, command):
  233. if command.argc >= 1:
  234. try:
  235. date = datetime.strptime(' '.join(command.args), '%d.%m.%Y')
  236. except ValueError:
  237. await event.reply('Дата не может быть распознана. Пожалуйста, введите свой день рождения в следующем формате: 01.01.1970 (день, месяц, год).')
  238. return
  239. if date >= datetime.now():
  240. await event.reply('День рождения не может быть в будущем...')
  241. return
  242. if await add_or_update_birthday(get_peer_id(event.peer_id), event.sender, date):
  243. await event.reply('День рождения успешно добавлен!!!')
  244. else:
  245. await event.reply('День рождения успешно обновлён!!!')
  246. return
  247. birthdays = await get_birthdays(get_peer_id(event.peer_id))
  248. if not birthdays:
  249. await event.reply('Пока пусто...')
  250. return
  251. birthdays = map(lambda birthday: (birthday.user_id, calculate_age(birthday.date)), birthdays)
  252. birthdays = sorted(birthdays, key=lambda birthday: birthday[1].days_until)
  253. birthdays_list = ''
  254. for user_id, age in birthdays:
  255. try:
  256. entity = await bot.get_entity(user_id)
  257. except ValueError:
  258. await bot.get_participants(await event.get_chat())
  259. entity = await bot.get_entity(user_id)
  260. birthdays_list += get_user_name(entity)
  261. birthdays_list += ' — '
  262. birthdays_list += age.date_string
  263. if age.days_until < 1:
  264. birthdays_list += f' (сегодня исполняется {get_word_for("год", age.age)})\n'
  265. else:
  266. birthdays_list += f' (через {get_word_for("день", age.days_until)} исполнится {get_word_for("год", age.age)}; сейчас {get_word_for("год", age.age_now)})\n'
  267. await event.reply(f'Дни рождения:\n\n{birthdays_list}')
  268. async def vpn_handler(bot, event, command):
  269. if command.argc < 1:
  270. await event.reply(f'Пожалуйста, укажите имя сервера! Доступные сервера: {await list_servers()}.')
  271. return
  272. try:
  273. ip = await get_server_ip(command.args[0])
  274. except SyntaxError:
  275. await event.reply('Недопустимое имя сервера!!')
  276. return
  277. except IndexError:
  278. await event.reply('Сервер с таким именем не найден!')
  279. return
  280. if event.sender is None:
  281. sender_id = event.peer_id.channel_id
  282. else:
  283. sender_id = event.sender.id
  284. async with ClientSession() as session:
  285. try:
  286. async with session.post(
  287. f'http://{ip}:9217/api/obtain',
  288. data={'user_id': sender_id}
  289. ) as resp:
  290. data = await resp.json()
  291. profile = data['profile']
  292. except:
  293. await event.reply('Произошла ошибка при попытке обращения к API сервера… :(')
  294. return
  295. try:
  296. await bot.send_message(
  297. await bot.get_entity(sender_id),
  298. f'Ваш файл конфигурации WireGuard (сервер {command.args[0]}):\n\n```{profile}```'
  299. )
  300. except:
  301. await event.reply('Произошла ошибка при отправке файла конфигурации в Ваши личные сообщения… :с')
  302. return
  303. await event.reply('Готово!!~~ Файл конфигурации WireGuard отправлен в Ваши личные сообщения!')
  304. async def sylvy_handler(bot, event, command):
  305. if command.argc < 1:
  306. await event.reply('Пожалуйста, не оставляйте ввод пустым!')
  307. return
  308. async with ClientSession() as session:
  309. try:
  310. async with session.post(
  311. f'https://sylvy-engine.txlyre.website/api/computeSingle',
  312. json={'program': command.args_string, 'stringify': True}
  313. ) as resp:
  314. data = await resp.json()
  315. if data['status'] != 'ok':
  316. await event.reply(f'Ошибка API Sylvy: {data["data"]["message"]}')
  317. return
  318. result = data['data']['result']
  319. except:
  320. await event.reply('Произошла ошибка при попытке обращения к API Sylvy… :(')
  321. return
  322. if result['status'] != 'SUCCESS':
  323. await event.reply(f'Ошибка исполнения!!!\n\n```{result["stdout"]}```')
  324. return
  325. if result['plots']:
  326. plots = []
  327. for plot in result['plots']:
  328. buffer = BytesIO()
  329. svg2png(
  330. bytestring=plot,
  331. write_to=buffer
  332. )
  333. plots.append(buffer.getvalue())
  334. await bot.send_file(
  335. event.peer_id,
  336. file=plots,
  337. reply_to=event
  338. )
  339. return
  340. text = ''
  341. if result['stdout']:
  342. text += result['stdout']
  343. text += '\n'
  344. text += result['output']
  345. await event.reply(f'```{text}```')
  346. COMMANDS = {
  347. 'newadmin': Handler(newadmin_handler, is_restricted=True),
  348. 'deladmin': Handler(deladmin_handler, is_restricted=True),
  349. 'newaction': Handler(newaction_handler, is_restricted=True),
  350. 'delaction': Handler(delaction_handler, is_restricted=True),
  351. 'addgif': Handler(addgif_handler, is_restricted=True),
  352. 'addserver': Handler(addserver_handler, is_restricted=True),
  353. 'delserver': Handler(delserver_handler, is_restricted=True),
  354. 'allow': Handler(allow_handler, is_restricted=True),
  355. 'disallow': Handler(disallow_handler, is_restricted=True),
  356. 'save': Handler(save_handler),
  357. 'bday': Handler(bday_handler),
  358. 'vpn': Handler(vpn_handler),
  359. 'sylvy': Handler(sylvy_handler),
  360. }