commands.py 12 KB

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