utils.py 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203
  1. from uuid import uuid4
  2. from enum import IntEnum
  3. from datetime import datetime
  4. from collections import namedtuple
  5. from telethon.utils import get_display_name
  6. from telethon.tl.types import (
  7. MessageEntityBold,
  8. MessageEntityItalic,
  9. MessageEntityStrike,
  10. MessageEntityCode,
  11. MessageEntityPre,
  12. MessageEntitySpoiler,
  13. MessageEntityUnderline,
  14. MessageEntityUrl
  15. )
  16. Command = namedtuple(
  17. 'Command',
  18. 'name argc args args_string'
  19. )
  20. Age = namedtuple(
  21. 'Age',
  22. 'days_until age age_now date_string'
  23. )
  24. class Kind(IntEnum):
  25. CANNOT_APPLY_TO_SELF = 0
  26. CAN_APPLY_TO_SELF = 1
  27. NO_TARGET = 2
  28. class WordKind(IntEnum):
  29. FIRST = 0 # «год» / «день».
  30. SECOND = 1 # «лет» / «дней».
  31. THIRD = 2 # «года» / «дня».
  32. WORDS_TABLE = {
  33. 'год': ('год', 'лет', 'года'),
  34. 'день': ('день', 'дней', 'дня')
  35. }
  36. def parse_command(text):
  37. text = text.strip()
  38. if not text.startswith('/'):
  39. raise ValueError
  40. text = text.split(' ')
  41. if len(text) < 1:
  42. raise ValueError
  43. command = text[0][1:].lower()
  44. if '@' in command:
  45. command = command.split('@')
  46. command = command[0]
  47. args = text[1:]
  48. argc = len(args)
  49. args_string = ' '.join(args)
  50. return Command(
  51. name=command,
  52. argc=argc,
  53. args=args,
  54. args_string=args_string
  55. )
  56. def get_user_name(user):
  57. full_name = get_display_name(user)
  58. if not full_name:
  59. full_name = user.username
  60. if not full_name:
  61. full_name = '?'
  62. return full_name
  63. def get_link_to_user(user):
  64. full_name = get_user_name(user)
  65. if user.username:
  66. return f'[{full_name}](@{user.username})'
  67. return f'[{full_name}](tg://user?id={user.id})'
  68. def is_valid_name(name):
  69. return name.isidentifier()
  70. def make_temporary_filename(ext):
  71. uid = uuid4().hex
  72. return f'tmp_{uid}.{ext}'
  73. def parse_kind(kind):
  74. kind = int(kind)
  75. if kind < 0 or kind > 2:
  76. raise ValueError
  77. return kind
  78. # Bruh?
  79. def get_word_kind(number):
  80. if number == 0 or 5 <= number <= 20:
  81. return WordKind.SECOND
  82. last_digit = number % 10
  83. if last_digit == 0:
  84. return WordKind.SECOND
  85. if last_digit == 1:
  86. return WordKind.FIRST
  87. if 5 <= last_digit <= 9:
  88. return WordKind.SECOND
  89. return WordKind.THIRD
  90. def get_word_for(word, number):
  91. return f'{number} {WORDS_TABLE[word][get_word_kind(number)]}'
  92. def calculate_age(date):
  93. now = datetime.now().date()
  94. birthday_date = date.replace(year=now.year)
  95. if now > birthday_date:
  96. birthday_date = date.replace(year=now.year + 1)
  97. delta = birthday_date - now
  98. age = (birthday_date - date).days // 365
  99. age_now = age - 1
  100. return Age(
  101. days_until=delta.days,
  102. age=age,
  103. age_now=age_now,
  104. date_string=date.strftime('%d.%m.%Y')
  105. )
  106. DELIMITERS = {
  107. MessageEntityBold: '**',
  108. MessageEntityItalic: '__',
  109. MessageEntityStrike: '~~',
  110. MessageEntityCode: '`',
  111. MessageEntityPre: '`',
  112. MessageEntityUnderline: '\ue000',
  113. MessageEntitySpoiler: '\ue001',
  114. MessageEntityUrl: '\ue002',
  115. }
  116. class LookupTable:
  117. def __init__(self):
  118. self._start = {}
  119. self._end = {}
  120. def insert(self, entity):
  121. delimiter = DELIMITERS.get(type(entity))
  122. if not delimiter:
  123. return
  124. start = entity.offset
  125. end = entity.offset + entity.length
  126. if start in self._start:
  127. self._start[start].append(delimiter)
  128. else:
  129. self._start[start] = [delimiter]
  130. if end in self._end:
  131. self._end[end].insert(0, delimiter)
  132. else:
  133. self._end[end] = [delimiter]
  134. def _lookup(self, position):
  135. if position in self._start:
  136. return ''.join(self._start[position])
  137. elif position in self._end:
  138. return ''.join(self._end[position])
  139. def process(self, text):
  140. text = list(text) + ['']
  141. result = ''
  142. for position, character in zip(range(len(text)), text):
  143. delimiter = self._lookup(position)
  144. if delimiter:
  145. result += delimiter
  146. result += character
  147. return result
  148. def unparse(text, entities):
  149. table = LookupTable()
  150. for entity in entities:
  151. table.insert(entity)
  152. return table.process(text)