commands.py 7.9 KB

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