rand.py 9.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355
  1. import re
  2. import random
  3. import struct
  4. import operator
  5. import aiohttp
  6. async def _pyrandom(min_val, max_val, is_dice=False):
  7. if is_dice:
  8. return [random.randint(1, max_val) for _ in range(min_val)]
  9. return random.randint(min_val, max_val)
  10. async def _get(url):
  11. async with aiohttp.ClientSession() as session:
  12. async with session.get(url) as resp:
  13. return await resp.read()
  14. async def generate_unbiased_numbers(data, min_val, max_val, count):
  15. mod = max_val - min_val + 1
  16. max_acceptable = (1 << 64) // mod * mod
  17. numbers = []
  18. data_len = len(data)
  19. bytes_needed = count * 8
  20. if data_len < bytes_needed:
  21. return None
  22. for i in range(count):
  23. chunk = data[i * 8 : (i + 1) * 8]
  24. number = struct.unpack("<Q", chunk)[0]
  25. while number >= max_acceptable:
  26. number = number >> 1
  27. numbers.append((number % mod) + min_val)
  28. return numbers
  29. async def generate(source, min_val, max_val, count=1):
  30. total_bytes = count * 8
  31. try:
  32. data = await _get(f"{source}{total_bytes}")
  33. except Exception:
  34. data = b""
  35. if len(data) >= total_bytes:
  36. result = await generate_unbiased_numbers(data, min_val, max_val, count)
  37. if result is not None:
  38. return result
  39. if count == 1:
  40. return [await _pyrandom(min_val, max_val)]
  41. return await _pyrandom(min_val, max_val, is_dice=True)
  42. async def _trng_yebisu(min_val, max_val, is_dice=False):
  43. count = min_val if is_dice else 1
  44. actual_min = 1 if is_dice else min_val
  45. actual_max = max_val if is_dice else max_val
  46. numbers = await generate(
  47. "https://yebi.su/api/pool?count=", actual_min, actual_max, count
  48. )
  49. return numbers if is_dice else numbers[0]
  50. async def rolldices(count, sides):
  51. try:
  52. return await _trng_yebisu(count, sides, is_dice=True)
  53. except Exception:
  54. return await _pyrandom(count, sides, is_dice=True)
  55. async def _roll(count, sides):
  56. if count <= 0:
  57. raise ValueError("Количество костей должно быть больше нуля.")
  58. if sides <= 0:
  59. raise ValueError("Количество сторон должно быть больше нуля.")
  60. return await rolldices(count, sides)
  61. OPS = {
  62. "+": operator.add,
  63. "-": operator.sub,
  64. "*": operator.mul,
  65. "/": operator.truediv,
  66. "%": operator.mod,
  67. "^": operator.pow,
  68. }
  69. OPS_KEYS = "".join(OPS.keys()).replace("-", r"\-")
  70. T_NAME = re.compile(r"([abce-zа-ге-йл-яA-ZА-Я]+)")
  71. T_COLON = re.compile(r"(:)")
  72. T_DICE = re.compile(r"(d|д|к)")
  73. T_MINUS = re.compile(r"(-)")
  74. T_OP = re.compile(f"([{OPS_KEYS}])")
  75. T_DIGIT = re.compile(r"(\d+)")
  76. T_OPEN_PAREN = re.compile(r"(\()")
  77. T_CLOSE_PAREN = re.compile(r"(\))")
  78. T_WS = re.compile(r"([ \t\r\n]+)")
  79. TOKEN_NAMES = {
  80. T_NAME: "имя",
  81. T_COLON: "двоеточие",
  82. T_DICE: "кость",
  83. T_OP: "оператор",
  84. T_DIGIT: "число",
  85. T_OPEN_PAREN: "открывающая скобка",
  86. T_CLOSE_PAREN: "закрывающая скобка",
  87. }
  88. class Value:
  89. def __init__(self, value):
  90. if not isinstance(value, list):
  91. value = [int(value)]
  92. self.value = value
  93. def __int__(self):
  94. return sum(map(int, self.value))
  95. def __repr__(self):
  96. return str(self.value[0] if len(self.value) == 1 else self.value)
  97. def __iter__(self):
  98. return iter(self.value)
  99. def __next__(self):
  100. return next(self.value)
  101. def __index__(self, index):
  102. return self.value[index]
  103. def __len__(self):
  104. return len(self.value)
  105. def __eq__(self, other):
  106. if not isinstance(other, Value):
  107. return False
  108. return self.value == other.value
  109. def apply(self, what, *args):
  110. args = list(map(int, args))
  111. return Value(what(int(self), *args))
  112. class Dices:
  113. def __init__(self, text):
  114. self.text = text.strip()
  115. self.position = 0
  116. self.names = {}
  117. self.rolls = []
  118. self.result = None
  119. self._rolls = []
  120. def __repr__(self):
  121. if self.result:
  122. buffer = ""
  123. for count, sides, roll in self._rolls:
  124. buffer += f"{'' if count == 1 else count}d{sides}: {roll}\n"
  125. for roll, result in zip(self.rolls, self.result):
  126. count, sides, roll = roll
  127. buffer += f"{'' if count == 1 else count}d{sides}: "
  128. roll_sum = int(roll)
  129. result_sum = int(result)
  130. if result_sum == roll_sum:
  131. results = ", ".join(map(str, result))
  132. if "," in results:
  133. results = f"[{results}]"
  134. buffer += results
  135. if "," in results:
  136. buffer += f" ({result_sum})"
  137. buffer += "\n"
  138. else:
  139. difference = result_sum - roll_sum
  140. results = f"{'' if len(roll) == 1 else f'{roll_sum} -> '}{roll_sum}{'' if difference < 0 else '+'}{difference}"
  141. buffer += f"{roll} -> {results} ({int(result)})\n"
  142. return f"{self.text}\n{buffer}= {int(self.result)}"
  143. return self.text
  144. def _skip_ws(self):
  145. match = T_WS.match(self.text, self.position)
  146. if match:
  147. self.position += len(match.group(0))
  148. def _done(self):
  149. self._skip_ws()
  150. return self.position >= len(self.text)
  151. def _match(self, what, skip_ws=True):
  152. if skip_ws:
  153. self._skip_ws()
  154. match = what.match(self.text, self.position)
  155. if match:
  156. self.position += len(match.group(0))
  157. return match.groups()
  158. def _expected(self, expected):
  159. raise SyntaxError(
  160. f"Неожиданный ввод на позиции `#{self.position + 1}`: ожидалось: `{expected}`."
  161. )
  162. def _expect(self, what):
  163. match = self._match(what)
  164. if not match:
  165. self._expected(TOKEN_NAMES[what])
  166. return match
  167. async def _parse_dice(self, left=1):
  168. if self._match(T_OPEN_PAREN):
  169. right = await self._parse_expr()
  170. self._expect(T_CLOSE_PAREN)
  171. else:
  172. right = await self._parse_atom()
  173. left = int(left)
  174. right = int(right)
  175. if left > 1000 or right > 1000:
  176. raise SyntaxError("Слишком длинное число.")
  177. roll = Value(await _roll(left, right))
  178. self._rolls.append((left, right, roll))
  179. return roll
  180. async def _parse_atom(self):
  181. if self._match(T_OPEN_PAREN):
  182. expr = await self._parse_expr()
  183. self._expect(T_CLOSE_PAREN)
  184. if self._match(T_DICE, skip_ws=False):
  185. return await self._parse_dice(expr)
  186. return expr
  187. elif self._match(T_MINUS):
  188. value = await self._parse_atom()
  189. return value.apply(operator.neg)
  190. elif match := self._match(T_DIGIT):
  191. try:
  192. left = int(match[0])
  193. except ValueError:
  194. raise SyntaxError("Слишком длинное число.")
  195. if match := self._match(T_DICE, skip_ws=False):
  196. return await self._parse_dice(left)
  197. return Value(left)
  198. elif self._match(T_DICE):
  199. return await self._parse_dice()
  200. elif match := self._match(T_NAME):
  201. name = match[0].upper()
  202. if name not in self.names:
  203. raise NameError(f"Неизвестная переменная: `{match[0]}`.")
  204. expr = self.names[name]
  205. if self._match(T_DICE, skip_ws=False):
  206. return await self._parse_dice(expr)
  207. return expr
  208. self._expected("число, кость или переменная")
  209. async def _parse_expr(self):
  210. left = await self._parse_atom()
  211. if op := self._match(T_OP):
  212. op = OPS[op[0]]
  213. right = await self._parse_expr()
  214. left = left.apply(op, right)
  215. elif self._match(T_COLON):
  216. right = self._expect(T_NAME)[0].upper()
  217. self.names[right] = left
  218. return left
  219. async def _parse_exprs(self):
  220. exprs = []
  221. while not self._done():
  222. if len(exprs) >= 10:
  223. raise SyntaxError("Слишком длинное число.")
  224. rolls_count = len(self._rolls)
  225. expr = await self._parse_expr()
  226. if len(self._rolls) == rolls_count:
  227. raise SyntaxError("Выражение не содержит бросков.")
  228. self.rolls.append(self._rolls.pop(-1))
  229. exprs.append(expr)
  230. if not exprs:
  231. raise SyntaxError("Выражение не должно быть пустым.")
  232. return Value(exprs)
  233. async def roll(self, vars={}):
  234. self.names = {str(k).upper(): Value(vars[k]) for k in vars}
  235. self.position = 0
  236. self.rolls = []
  237. self._rolls = []
  238. self.result = await self._parse_exprs()
  239. return self
  240. async def roll_dices(dices, vars={}):
  241. dices = Dices(dices)
  242. try:
  243. await dices.roll(vars=vars)
  244. except (ValueError, SyntaxError, NameError) as e:
  245. return str(e)
  246. except ZeroDivisionError:
  247. raise "Попытка деления на ноль."
  248. return str(dices)