commands.py 8.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301
  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 remove
  10. from aiofiles.os import path
  11. from emoji import is_emoji
  12. from actions import (
  13. find_action,
  14. create_action,
  15. delete_action,
  16. add_gif,
  17. add_sticker,
  18. add_admin,
  19. delete_admin,
  20. add_or_update_birthday,
  21. get_birthdays
  22. )
  23. from utils import (
  24. WORDS_TABLE,
  25. make_temporary_filename,
  26. make_cache_filename,
  27. parse_kind,
  28. get_user_name,
  29. get_word_for,
  30. calculate_age,
  31. unparse
  32. )
  33. class Handler:
  34. def __init__(self, handler, is_restricted=False):
  35. self.handler = handler
  36. self.is_restricted = is_restricted
  37. async def newadmin_handler(bot, event, command):
  38. if command.argc < 1:
  39. await event.reply('Пожалуйста, укажите пользователя!')
  40. return
  41. try:
  42. target = await bot.get_entity(command.args[0])
  43. except ValueError:
  44. await event.reply('Недопустимое имя пользователя!')
  45. return
  46. try:
  47. await add_admin(target)
  48. except IntegrityError:
  49. await event.reply('Данный пользователь уже является администратором!')
  50. return
  51. await event.reply('Готово!~~')
  52. async def deladmin_handler(bot, event, command):
  53. if command.argc < 1:
  54. await event.reply('Пожалуйста, укажите пользователя!')
  55. return
  56. try:
  57. target = await bot.get_entity(command.args[0])
  58. except ValueError:
  59. await event.reply('Недопустимое имя пользователя!')
  60. return
  61. try:
  62. await delete_admin(target)
  63. except IndexError:
  64. await event.reply('Данный пользователь не является администратором!')
  65. return
  66. await event.reply('Готово!~~')
  67. async def newaction_handler(bot, event, command):
  68. if command.argc < 3:
  69. await event.reply('Пожалуйста, укажите тип, имя и шаблон действия!')
  70. return
  71. try:
  72. kind = parse_kind(command.args[0])
  73. except ValueError:
  74. await event.reply('Неверный тип действия!!!')
  75. return
  76. try:
  77. await create_action(command.args[1], ' '.join(command.args[2:]), kind)
  78. except SyntaxError:
  79. await event.reply('Недопустимое имя действия!!!')
  80. return
  81. except IntegrityError:
  82. await event.reply('Действие с таким названием уже существует!')
  83. return
  84. await event.reply('Действие создано!')
  85. async def delaction_handler(bot, event, command):
  86. if command.argc < 1:
  87. await event.reply('Пожалуйста, укажите имя действия!')
  88. return
  89. try:
  90. await delete_action(command.args[0])
  91. except SyntaxError:
  92. await event.reply('Недопустимое имя действия!!!')
  93. return
  94. except NameError:
  95. await event.reply('Действия с таким названием не существует!')
  96. return
  97. await event.reply('Действие удалено!')
  98. async def addgif_handler(bot, event, command):
  99. if command.argc < 1:
  100. await event.reply('Пожалуйста, укажите имя действия!')
  101. return
  102. gif = await event.get_reply_message()
  103. if not gif or not gif.gif:
  104. await event.reply('Пожалуйста, добавьте GIF!')
  105. return
  106. try:
  107. action = await find_action(command.args[0])
  108. await add_gif(action, gif.file.id)
  109. except SyntaxError:
  110. await event.reply('Недопустимое имя действия!!!')
  111. return
  112. except NameError:
  113. await event.reply('Нет такого действия!')
  114. return
  115. await event.reply('Готово!~~')
  116. # Very, very, VERY evil code...
  117. async def make_message_shot(bot, message):
  118. if message.sender is None:
  119. sender_id = message.peer_id.channel_id
  120. full_name = await bot.get_entity(sender_id)
  121. full_name = full_name.title
  122. else:
  123. sender_id = message.sender.id
  124. full_name = get_display_name(message.sender)
  125. output_path = make_temporary_filename('png')
  126. avatar_path = await make_cache_filename(sender_id, 'png')
  127. if not await path.isfile(avatar_path):
  128. await bot.download_profile_photo(sender_id, file=avatar_path)
  129. # TO-DO: make it better.
  130. mproc = await create_subprocess_shell(
  131. f'mogrify -format png {avatar_path}'
  132. )
  133. await mproc.communicate()
  134. data = bytes()
  135. data += pack('I', len(output_path))
  136. data += bytes(output_path, encoding='ASCII')
  137. data += pack('I', len(avatar_path))
  138. data += bytes(avatar_path, encoding='ASCII')
  139. username = bytes(full_name, encoding='UTF-8')
  140. data += pack('I', len(username))
  141. data += username
  142. data += pack('I', sender_id % 7)
  143. text = bytes(unparse(message.raw_text, [entity for entity, _ in message.get_entities_text()]), encoding='UTF-8')
  144. data += pack('I', len(text))
  145. data += text
  146. proc = await create_subprocess_shell(
  147. './makeshot/makeshot',
  148. stdin=PIPE
  149. )
  150. await proc.communicate(input=data)
  151. pproc = await create_subprocess_shell(
  152. f'pngcrush -reduce -ow {output_path} {output_path}'
  153. )
  154. await pproc.communicate()
  155. return output_path
  156. async def save_handler(bot, event, command):
  157. message = await event.get_reply_message()
  158. if not message:
  159. await event.reply('Пожалуйста, укажите сообщение для сохранения!')
  160. return
  161. emoji = '⚡'
  162. if command.argc >= 1:
  163. emoji = command.args[0]
  164. if len(emoji) not in range(1, 6)\
  165. or not all(map(is_emoji, emoji)):
  166. await event.reply('Указан некорректный эмодзи!!!')
  167. return
  168. path = await make_message_shot(bot, message)
  169. try:
  170. file = await add_sticker(bot, path, emoji)
  171. await bot.send_file(
  172. message.peer_id,
  173. file=file,
  174. reply_to=message
  175. )
  176. finally:
  177. await remove(path)
  178. async def bday_handler(bot, event, command):
  179. if command.argc >= 1:
  180. try:
  181. date = datetime.strptime(' '.join(command.args), '%d.%m.%Y')
  182. except ValueError:
  183. await event.reply('Дата не может быть распознана. Пожалуйста, введите свой день рождения в следующем формате: 01.01.1970 (день, месяц, год).')
  184. return
  185. if date >= datetime.now():
  186. await event.reply('День рождения не может быть в будущем...')
  187. return
  188. if await add_or_update_birthday(get_peer_id(event.peer_id), event.sender, date):
  189. await event.reply('День рождения успешно добавлен!!!')
  190. else:
  191. await event.reply('День рождения успешно обновлён!!!')
  192. return
  193. birthdays = await get_birthdays(get_peer_id(event.peer_id))
  194. if not birthdays:
  195. await event.reply('Пока пусто...')
  196. return
  197. birthdays = map(lambda birthday: (birthday.user_id, calculate_age(birthday.date)), birthdays)
  198. birthdays = sorted(birthdays, key=lambda birthday: birthday[1].days_until)
  199. birthdays_list = ''
  200. for user_id, age in birthdays:
  201. try:
  202. entity = await bot.get_entity(user_id)
  203. except ValueError:
  204. await bot.get_participants(await event.get_chat())
  205. entity = await bot.get_entity(user_id)
  206. birthdays_list += get_user_name(entity)
  207. birthdays_list += ' — '
  208. birthdays_list += age.date_string
  209. if age.days_until < 1:
  210. birthdays_list += f' (сегодня исполняется {get_word_for("год", age.age)})\n'
  211. else:
  212. birthdays_list += f' (через {get_word_for("день", age.days_until)} исполнится {get_word_for("год", age.age)}; сейчас {get_word_for("год", age.age_now)})\n'
  213. await event.reply(f'Дни рождения:\n\n{birthdays_list}')
  214. COMMANDS = {
  215. 'newadmin': Handler(newadmin_handler, is_restricted=True),
  216. 'deladmin': Handler(deladmin_handler, is_restricted=True),
  217. 'newaction': Handler(newaction_handler, is_restricted=True),
  218. 'delaction': Handler(delaction_handler, is_restricted=True),
  219. 'addgif': Handler(addgif_handler, is_restricted=True),
  220. 'save': Handler(save_handler),
  221. 'bday': Handler(bday_handler)
  222. }