commands.py 15 KB

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