Queer European MD passionate about IT

utilities.py 46 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653
  1. """Useful functions used by Davte when programming in python."""
  2. # Standard library modules
  3. import asyncio
  4. import collections
  5. import csv
  6. import datetime
  7. import inspect
  8. import io
  9. import json
  10. import logging
  11. import os
  12. import random
  13. import re
  14. import string
  15. import time
  16. from difflib import SequenceMatcher
  17. # Third party modules
  18. import aiohttp
  19. from aiohttp import web
  20. from bs4 import BeautifulSoup
  21. weekdays = collections.OrderedDict()
  22. weekdays[0] = {
  23. 'en': "Sunday",
  24. 'it': "Domenica",
  25. }
  26. weekdays[1] = {
  27. 'en': "Monday",
  28. 'it': "Lunedì",
  29. }
  30. weekdays[2] = {
  31. 'en': "Tuesday",
  32. 'it': "Martedì",
  33. }
  34. weekdays[3] = {
  35. 'en': "Wednesday",
  36. 'it': "Mercoledì",
  37. }
  38. weekdays[4] = {
  39. 'en': "Thursday",
  40. 'it': "Giovedì",
  41. }
  42. weekdays[5] = {
  43. 'en': "Friday",
  44. 'it': "Venerdì",
  45. }
  46. weekdays[6] = {
  47. 'en': "Saturday",
  48. 'it': "Sabato",
  49. }
  50. def sumif(iterable, condition):
  51. """Sum all `iterable` items matching `condition`."""
  52. return sum(
  53. filter(
  54. condition,
  55. iterable
  56. )
  57. )
  58. def markdown_check(text, symbols):
  59. """Check that all `symbols` occur an even number of times in `text`."""
  60. for s in symbols:
  61. if (len(text.replace(s, "")) - len(text)) % 2 != 0:
  62. return False
  63. return True
  64. def shorten_text(text, limit, symbol="[...]"):
  65. """Return a given text truncated at limit if longer than limit.
  66. On truncation, add symbol.
  67. """
  68. assert type(text) is str and type(symbol) is str and type(limit) is int
  69. if len(text) <= limit:
  70. return text
  71. return text[:limit-len(symbol)] + symbol
  72. def extract(text, starter=None, ender=None):
  73. """Return string in text between starter and ender.
  74. If starter is None, truncate at ender.
  75. """
  76. if starter and starter in text:
  77. text = text.partition(starter)[2]
  78. if ender:
  79. return text.partition(ender)[0]
  80. return text
  81. def make_button(text=None, callback_data='',
  82. prefix='', delimiter='|', data=[]):
  83. """Return a Telegram bot API-compliant button.
  84. callback_data can be either a ready-to-use string or a
  85. prefix + delimiter-joined data.
  86. If callback_data exceeds Telegram limits (currently 60 characters),
  87. it gets truncated at the last delimiter before that limit.
  88. If absent, text is the same as callback_data.
  89. """
  90. if len(data):
  91. callback_data += delimiter.join(map(str, data))
  92. callback_data = "{p}{c}".format(
  93. p=prefix,
  94. c=callback_data
  95. )
  96. if len(callback_data) > 60:
  97. callback_data = callback_data[:61]
  98. callback_data = callback_data[:-1-callback_data[::-1].find(delimiter)]
  99. if text is None:
  100. text = callback_data
  101. return dict(
  102. text=text,
  103. callback_data=callback_data
  104. )
  105. def mkbtn(x, y):
  106. """Backward compatibility.
  107. Warning: this function will be deprecated sooner or later,
  108. stop using it and migrate to make_button.
  109. """
  110. return make_button(text=x, callback_data=y)
  111. def make_lines_of_buttons(buttons, row_len=1):
  112. """Split `buttons` list in a list of lists having length = `row_len`."""
  113. return [
  114. buttons[i:i + row_len]
  115. for i in range(
  116. 0,
  117. len(buttons),
  118. row_len
  119. )
  120. ]
  121. def make_inline_keyboard(buttons, row_len=1):
  122. """Make a Telegram API compliant inline keyboard."""
  123. return dict(
  124. inline_keyboard=make_lines_of_buttons(
  125. buttons,
  126. row_len
  127. )
  128. )
  129. async def async_get(url, mode='json', **kwargs):
  130. """Make an async get request.
  131. `mode`s allowed:
  132. * html
  133. * json
  134. * string
  135. Additional **kwargs may be passed.
  136. """
  137. if 'mode' in kwargs:
  138. mode = kwargs['mode']
  139. del kwargs['mode']
  140. return await async_request(
  141. url,
  142. type='get',
  143. mode=mode,
  144. **kwargs
  145. )
  146. async def async_post(url, mode='html', **kwargs):
  147. """Make an async post request.
  148. `mode`s allowed:
  149. * html
  150. * json
  151. * string
  152. Additional **kwargs may be passed (will be converted to `data`).
  153. """
  154. return await async_request(
  155. url,
  156. type='post',
  157. mode=mode,
  158. **kwargs
  159. )
  160. async def async_request(url, type='get', mode='json', encoding=None, errors='strict',
  161. **kwargs):
  162. """Make an async html request.
  163. `types` allowed
  164. * get
  165. * post
  166. `mode`s allowed:
  167. * html
  168. * json
  169. * string
  170. * picture
  171. Additional **kwargs may be passed.
  172. """
  173. try:
  174. async with aiohttp.ClientSession() as s:
  175. async with (
  176. s.get(url, timeout=30)
  177. if type == 'get'
  178. else s.post(url, timeout=30, data=kwargs)
  179. ) as r:
  180. if mode in ['html', 'json', 'string']:
  181. result = await r.text(encoding=encoding, errors=errors)
  182. else:
  183. result = await r.read()
  184. if encoding is not None:
  185. result = result.decode(encoding)
  186. except Exception as e:
  187. logging.error(
  188. 'Error making async request to {}:\n{}'.format(
  189. url,
  190. e
  191. ),
  192. exc_info=False
  193. ) # Set exc_info=True to debug
  194. return e
  195. if mode == 'json':
  196. try:
  197. result = json.loads(
  198. result
  199. )
  200. except json.decoder.JSONDecodeError:
  201. result = {}
  202. elif mode == 'html':
  203. result = BeautifulSoup(result, "html.parser")
  204. elif mode == 'string':
  205. result = result
  206. return result
  207. def json_read(file_, default={}, encoding='utf-8', **kwargs):
  208. """Return json parsing of `file_`, or `default` if file does not exist.
  209. `encoding` refers to how the file should be read.
  210. `kwargs` will be passed to json.load()
  211. """
  212. if not os.path.isfile(file_):
  213. return default
  214. with open(file_, "r", encoding=encoding) as f:
  215. return json.load(f, **kwargs)
  216. def json_write(what, file_, encoding='utf-8', **kwargs):
  217. """Store `what` in json `file_`.
  218. `encoding` refers to how the file should be written.
  219. `kwargs` will be passed to json.dump()
  220. """
  221. with open(file_, "w", encoding=encoding) as f:
  222. return json.dump(what, f, indent=4, **kwargs)
  223. def csv_read(file_, default=[], encoding='utf-8',
  224. delimiter=',', quotechar='"', **kwargs):
  225. """Return csv parsing of `file_`, or `default` if file does not exist.
  226. `encoding` refers to how the file should be read.
  227. `delimiter` is the separator of fields.
  228. `quotechar` is the string delimiter.
  229. `kwargs` will be passed to csv.reader()
  230. """
  231. if not os.path.isfile(file_):
  232. return default
  233. result = []
  234. keys = []
  235. with open(file_, newline='', encoding=encoding) as csv_file:
  236. csv_reader = csv.reader(
  237. csv_file,
  238. delimiter=delimiter,
  239. quotechar=quotechar,
  240. **kwargs
  241. )
  242. for row in csv_reader:
  243. if not keys:
  244. keys = row
  245. continue
  246. item = collections.OrderedDict()
  247. for key, val in zip(keys, row):
  248. item[key] = val
  249. result.append(item)
  250. return result
  251. def csv_write(info=[], file_='output.csv', encoding='utf-8',
  252. delimiter=',', quotechar='"', **kwargs):
  253. """Store `info` in CSV `file_`.
  254. `encoding` refers to how the file should be read.
  255. `delimiter` is the separator of fields.
  256. `quotechar` is the string delimiter.
  257. `encoding` refers to how the file should be written.
  258. `kwargs` will be passed to csv.writer()
  259. """
  260. assert (
  261. type(info) is list
  262. and len(info) > 0
  263. ), "info must be a non-empty list"
  264. assert all(
  265. isinstance(row, dict)
  266. for row in info
  267. ), "Rows must be dictionaries!"
  268. with open(file_, 'w', newline='', encoding=encoding) as csv_file:
  269. csv_writer = csv.writer(
  270. csv_file,
  271. delimiter=delimiter,
  272. quotechar=quotechar,
  273. **kwargs
  274. )
  275. csv_writer.writerow(info[0].keys())
  276. for row in info:
  277. csv_writer.writerow(row.values())
  278. return
  279. class MyOD(collections.OrderedDict):
  280. """Subclass of OrderedDict.
  281. It features `get_by_val` and `get_by_key_val` methods.
  282. """
  283. def __init__(self, *args, **kwargs):
  284. """Return a MyOD instance."""
  285. super().__init__(*args, **kwargs)
  286. self._anti_list_casesensitive = None
  287. self._anti_list_caseinsensitive = None
  288. @property
  289. def anti_list_casesensitive(self):
  290. """Case-sensitive reverse dictionary.
  291. Keys and values are swapped.
  292. """
  293. if not self._anti_list_casesensitive:
  294. self._anti_list_casesensitive = {
  295. val: key
  296. for key, val in self.items()
  297. }
  298. return self._anti_list_casesensitive
  299. @property
  300. def anti_list_caseinsensitive(self):
  301. """Case-sensitive reverse dictionary.
  302. Keys and values are swapped and lowered.
  303. """
  304. if not self._anti_list_caseinsensitive:
  305. self._anti_list_caseinsensitive = {
  306. (val.lower() if type(val) is str else val): key
  307. for key, val in self.items()
  308. }
  309. return self._anti_list_caseinsensitive
  310. def get_by_val(self, val, case_sensitive=True):
  311. """Get key pointing to given val.
  312. Can be case-sensitive or insensitive.
  313. MyOD[key] = val <-- MyOD.get(key) = val <--> MyOD.get_by_val(val) = key
  314. """
  315. return (
  316. self.anti_list_casesensitive
  317. if case_sensitive
  318. else self.anti_list_caseinsensitive
  319. )[val]
  320. def get_by_key_val(self, key, val,
  321. case_sensitive=True, return_value=False):
  322. """Get key (or val) of a dict-like object having key == val.
  323. Perform case-sensitive or insensitive search.
  324. """
  325. for x, y in self.items():
  326. if (
  327. (
  328. y[key] == val
  329. and case_sensitive
  330. ) or (
  331. y[key].lower() == val.lower()
  332. and not case_sensitive
  333. )
  334. ):
  335. return (
  336. y if return_value
  337. else x
  338. )
  339. return None
  340. def line_drawing_unordered_list(l):
  341. """Draw an old-fashioned unordered list.
  342. Unorderd list example
  343. ├ An element
  344. ├ Another element
  345. └Last element
  346. """
  347. result = ""
  348. if l:
  349. for x in l[:-1]:
  350. result += "├ {}\n".format(x)
  351. result += "└ {}".format(l[-1])
  352. return result
  353. def str_to_datetime(d):
  354. """Convert string to datetime.
  355. Dataset library often casts datetimes to str, this is a workaround.
  356. """
  357. if isinstance(d, datetime.datetime):
  358. return d
  359. return datetime.datetime.strptime(
  360. d,
  361. '%Y-%m-%d %H:%M:%S.%f'
  362. )
  363. def datetime_to_str(d):
  364. """Cast datetime to string."""
  365. if isinstance(d, str):
  366. d = str_to_datetime(d)
  367. if not isinstance(d, datetime.datetime):
  368. raise TypeError(
  369. 'Input of datetime_to_str function must be a datetime.datetime '
  370. 'object. Output is a str'
  371. )
  372. return '{:%Y-%m-%d %H:%M:%S.%f}'.format(d)
  373. class MyCounter():
  374. """Counter object, with a `lvl` method incrementing `n` property."""
  375. def __init__(self):
  376. """Initialize and get MyCounter instance."""
  377. self._n = 0
  378. return
  379. def lvl(self):
  380. """Increments and return self.n."""
  381. self._n += 1
  382. return self.n
  383. def reset(self):
  384. """Set self.n = 0."""
  385. self._n = 0
  386. return self.n
  387. def n(self):
  388. """Counter's value."""
  389. return self._n
  390. def wrapper(func, *args, **kwargs):
  391. """Wrap a function so that it can be later called with one argument."""
  392. def wrapped(update):
  393. return func(update, *args, **kwargs)
  394. return wrapped
  395. async def async_wrapper(coroutine, *args1, **kwargs1):
  396. """Wrap a `coroutine` so that it can be later awaited with more arguments.
  397. Set some of the arguments, let the coroutine be awaited with the rest of
  398. them later.
  399. The wrapped coroutine will always pass only supported parameters to
  400. `coroutine`.
  401. Example:
  402. ```
  403. import asyncio
  404. from davtelepot.utilities import async_wrapper
  405. async def printer(a, b, c, d):
  406. print(a, a+b, b+c, c+d)
  407. return
  408. async def main():
  409. my_coroutine = await async_wrapper(
  410. printer,
  411. c=3, d=2
  412. )
  413. await my_coroutine(a=1, b=5)
  414. asyncio.get_event_loop().run_until_complete(main())
  415. ```
  416. """
  417. async def wrapped_coroutine(*args2, bot=None, update=None, user_record=None, **kwargs2):
  418. # Update keyword arguments
  419. kwargs1.update(kwargs2)
  420. kwargs1['bot'] = bot
  421. kwargs1['update'] = update
  422. kwargs1['user_record'] = user_record
  423. # Pass only supported arguments
  424. kwargs = {
  425. name: argument
  426. for name, argument in kwargs1.items()
  427. if name in inspect.signature(
  428. coroutine
  429. ).parameters
  430. }
  431. return await coroutine(*args1, *args2, **kwargs)
  432. return wrapped_coroutine
  433. def forwarded(by=None):
  434. """Check that update is forwarded, optionally `by` someone in particular.
  435. Decorator: such decorated functions have effect only if update
  436. is forwarded from someone (you can specify `by` whom).
  437. """
  438. def is_forwarded_by(update, by):
  439. if 'forward_from' not in update:
  440. return False
  441. if by and update['forward_from']['id'] != by:
  442. return False
  443. return True
  444. def decorator(view_func):
  445. if asyncio.iscoroutinefunction(view_func):
  446. async def decorated(update):
  447. if is_forwarded_by(update, by):
  448. return await view_func(update)
  449. else:
  450. def decorated(update):
  451. if is_forwarded_by(update, by):
  452. return view_func(update)
  453. return decorated
  454. return decorator
  455. def chat_selective(chat_id=None):
  456. """Check that update comes from a chat, optionally having `chat_id`.
  457. Such decorated functions have effect only if update comes from
  458. a specific (if `chat_id` is given) or generic chat.
  459. """
  460. def check_function(update, chat_id):
  461. if 'chat' not in update:
  462. return False
  463. if chat_id:
  464. if update['chat']['id'] != chat_id:
  465. return False
  466. return True
  467. def decorator(view_func):
  468. if asyncio.iscoroutinefunction(view_func):
  469. async def decorated(update):
  470. if check_function(update, chat_id):
  471. return await view_func(update)
  472. else:
  473. def decorated(update):
  474. if check_function(update, chat_id):
  475. return view_func(update)
  476. return decorated
  477. return decorator
  478. async def sleep_until(when):
  479. """Sleep until now > `when`.
  480. `when` could be a datetime.datetime or a datetime.timedelta instance.
  481. """
  482. if not (
  483. isinstance(when, datetime.datetime)
  484. or isinstance(when, datetime.timedelta)
  485. ):
  486. raise TypeError(
  487. "sleep_until takes a datetime.datetime or datetime.timedelta "
  488. "object as argument!"
  489. )
  490. if isinstance(when, datetime.datetime):
  491. delta = when - datetime.datetime.now()
  492. elif isinstance(when, datetime.timedelta):
  493. delta = when
  494. if delta.days >= 0:
  495. await asyncio.sleep(
  496. delta.seconds
  497. )
  498. return
  499. async def wait_and_do(when, what, *args, **kwargs):
  500. """Sleep until `when`, then call `what` passing `args` and `kwargs`."""
  501. await sleep_until(when)
  502. return await what(*args, **kwargs)
  503. def get_csv_string(list_, delimiter=',', quotechar='"'):
  504. """Return a `delimiter`-delimited string of `list_` items.
  505. Wrap strings in `quotechar`s.
  506. """
  507. return delimiter.join(
  508. str(item) if type(item) is not str
  509. else '{q}{i}{q}'.format(
  510. i=item,
  511. q=quotechar
  512. )
  513. for item in list_
  514. )
  515. def case_accent_insensitive_sql(field):
  516. """Get a SQL string to perform a case- and accent-insensitive query.
  517. Given a `field`, return a part of SQL string necessary to perform
  518. a case- and accent-insensitive query.
  519. """
  520. replacements = [
  521. (' ', ''),
  522. ('à', 'a'),
  523. ('è', 'e'),
  524. ('é', 'e'),
  525. ('ì', 'i'),
  526. ('ò', 'o'),
  527. ('ù', 'u'),
  528. ]
  529. return "{r}LOWER({f}){w}".format(
  530. r="replace(".upper()*len(replacements),
  531. f=field,
  532. w=''.join(
  533. ", '{w[0]}', '{w[1]}')".format(w=w)
  534. for w in replacements
  535. )
  536. )
  537. # Italian definite articles.
  538. ARTICOLI = MyOD()
  539. ARTICOLI[1] = {
  540. 'ind': 'un',
  541. 'dets': 'il',
  542. 'detp': 'i',
  543. 'dess': 'l',
  544. 'desp': 'i'
  545. }
  546. ARTICOLI[2] = {
  547. 'ind': 'una',
  548. 'dets': 'la',
  549. 'detp': 'le',
  550. 'dess': 'lla',
  551. 'desp': 'lle'
  552. }
  553. ARTICOLI[3] = {
  554. 'ind': 'uno',
  555. 'dets': 'lo',
  556. 'detp': 'gli',
  557. 'dess': 'llo',
  558. 'desp': 'gli'
  559. }
  560. ARTICOLI[4] = {
  561. 'ind': 'un',
  562. 'dets': 'l\'',
  563. 'detp': 'gli',
  564. 'dess': 'll\'',
  565. 'desp': 'gli'
  566. }
  567. class Gettable():
  568. """Gettable objects can be retrieved from memory without being duplicated.
  569. Key is the primary key.
  570. Use classmethod get to instantiate (or retrieve) Gettable objects.
  571. Assign SubClass.instances = {}, otherwise Gettable.instances will
  572. contain SubClass objects.
  573. """
  574. instances = {}
  575. @classmethod
  576. def get(cls, key, *args, **kwargs):
  577. """Instantiate and/or retrieve Gettable object.
  578. SubClass.instances is searched if exists.
  579. Gettable.instances is searched otherwise.
  580. """
  581. if key not in cls.instances:
  582. cls.instances[key] = cls(key, *args, **kwargs)
  583. return cls.instances[key]
  584. class Confirmable():
  585. """Confirmable objects are provided with a confirm instance method.
  586. It evaluates True if it was called within self._confirm_timedelta,
  587. False otherwise.
  588. When it returns True, timer is reset.
  589. """
  590. CONFIRM_TIMEDELTA = datetime.timedelta(seconds=10)
  591. def __init__(self, confirm_timedelta=None):
  592. """Instantiate Confirmable instance.
  593. If `confirm_timedelta` is not passed,
  594. `self.__class__.CONFIRM_TIMEDELTA` is used as default.
  595. """
  596. if confirm_timedelta is None:
  597. confirm_timedelta = self.__class__.CONFIRM_TIMEDELTA
  598. self.set_confirm_timedelta(confirm_timedelta)
  599. self._confirm_datetimes = {}
  600. @property
  601. def confirm_timedelta(self):
  602. """Maximum timedelta between two calls of `confirm`."""
  603. return self._confirm_timedelta
  604. def confirm_datetime(self, who='unique'):
  605. """Get datetime of `who`'s last confirm.
  606. If `who` never called `confirm`, fake an expired call.
  607. """
  608. if who not in self._confirm_datetimes:
  609. self._confirm_datetimes[who] = (
  610. datetime.datetime.now()
  611. - 2*self.confirm_timedelta
  612. )
  613. return self._confirm_datetimes[who]
  614. def set_confirm_timedelta(self, confirm_timedelta):
  615. """Change self._confirm_timedelta."""
  616. if type(confirm_timedelta) is int:
  617. confirm_timedelta = datetime.timedelta(
  618. seconds=confirm_timedelta
  619. )
  620. assert isinstance(
  621. confirm_timedelta, datetime.timedelta
  622. ), "confirm_timedelta must be a datetime.timedelta instance!"
  623. self._confirm_timedelta = confirm_timedelta
  624. def confirm(self, who='unique'):
  625. """Return True if `confirm` was called by `who` recetly enough."""
  626. now = datetime.datetime.now()
  627. if now >= self.confirm_datetime(who) + self.confirm_timedelta:
  628. self._confirm_datetimes[who] = now
  629. return False
  630. self._confirm_datetimes[who] = now - 2*self.confirm_timedelta
  631. return True
  632. class HasBot():
  633. """Objects having a Bot subclass object as `.bot` attribute.
  634. HasBot objects have a .bot and .db properties for faster access.
  635. """
  636. bot = None
  637. @property
  638. def bot(self):
  639. """Class bot."""
  640. return self.__class__.bot
  641. @property
  642. def db(self):
  643. """Class bot db."""
  644. return self.bot.db
  645. @classmethod
  646. def set_bot(cls, bot):
  647. """Change class bot."""
  648. cls.bot = bot
  649. class CachedPage(Gettable):
  650. """Cache a webpage and return it during CACHE_TIME, otherwise refresh.
  651. Usage:
  652. cached_page = CachedPage.get(
  653. 'https://www.google.com',
  654. datetime.timedelta(seconds=30),
  655. **kwargs
  656. )
  657. page = await cached_page.get_page()
  658. """
  659. CACHE_TIME = datetime.timedelta(minutes=5)
  660. instances = {}
  661. def __init__(self, url, cache_time=None, **async_get_kwargs):
  662. """Instantiate CachedPage object.
  663. `url`: the URL to be cached
  664. `cache_time`: timedelta from last_update during which
  665. page will be cached
  666. `**kwargs` will be passed to async_get function
  667. """
  668. self._url = url
  669. if type(cache_time) is int:
  670. cache_time = datetime.timedelta(seconds=cache_time)
  671. if cache_time is None:
  672. cache_time = self.__class__.CACHE_TIME
  673. assert type(cache_time) is datetime.timedelta, (
  674. "Cache time must be a datetime.timedelta object!"
  675. )
  676. self._cache_time = cache_time
  677. self._page = None
  678. self._last_update = datetime.datetime.now() - self.cache_time
  679. self._async_get_kwargs = async_get_kwargs
  680. @property
  681. def url(self):
  682. """Get cached page url."""
  683. return self._url
  684. @property
  685. def cache_time(self):
  686. """Get cache time."""
  687. return self._cache_time
  688. @property
  689. def page(self):
  690. """Get webpage."""
  691. return self._page
  692. @property
  693. def last_update(self):
  694. """Get datetime of last update."""
  695. return self._last_update
  696. @property
  697. def async_get_kwargs(self):
  698. """Get async get request keyword arguments."""
  699. return self._async_get_kwargs
  700. @property
  701. def is_old(self):
  702. """Evaluate True if `chache_time` has passed since last update."""
  703. return datetime.datetime.now() > self.last_update + self.cache_time
  704. async def refresh(self):
  705. """Update cached webpage."""
  706. try:
  707. self._page = await async_get(self.url, **self.async_get_kwargs)
  708. self._last_update = datetime.datetime.now()
  709. return 0
  710. except Exception as e:
  711. self._page = None
  712. logging.error(
  713. '{e}'.format(
  714. e=e
  715. ),
  716. exc_info=False
  717. ) # Set exc_info=True to debug
  718. return 1
  719. return 1
  720. async def get_page(self):
  721. """Refresh if necessary and return webpage."""
  722. if self.is_old:
  723. await self.refresh()
  724. return self.page
  725. class Confirmator(Gettable, Confirmable):
  726. """Gettable Confirmable object."""
  727. instances = {}
  728. def __init__(self, key, *args, confirm_timedelta=None):
  729. """Call Confirmable.__init__ passing `confirm_timedelta`."""
  730. Confirmable.__init__(self, confirm_timedelta)
  731. def get_cleaned_text(update, bot=None, replace=[], strip='/ @'):
  732. """Clean `update`['text'] and return it.
  733. Strip `bot`.name and items to be `replace`d from the beginning of text.
  734. Strip `strip` characters from both ends.
  735. """
  736. if bot is not None:
  737. replace.append(
  738. '@{.name}'.format(
  739. bot
  740. )
  741. )
  742. text = update['text'].strip(strip)
  743. # Replace longer strings first
  744. for s in sorted(replace, key=len, reverse=True):
  745. while s and text.lower().startswith(s.lower()):
  746. text = text[len(s):]
  747. return text.strip(strip)
  748. def get_user(record, link_profile=True):
  749. """Get an HTML Telegram tag for user `record`."""
  750. if not record:
  751. return
  752. from_ = {key: val for key, val in record.items()}
  753. result = '{name}'
  754. if 'telegram_id' in from_:
  755. from_['id'] = from_['telegram_id']
  756. if (
  757. 'id' in from_
  758. and from_['id'] is not None
  759. and link_profile
  760. ):
  761. result = f"""<a href="tg://user?id={from_['id']}">{{name}}</a>"""
  762. if 'username' in from_ and from_['username']:
  763. result = result.format(
  764. name=from_['username']
  765. )
  766. elif (
  767. 'first_name' in from_
  768. and from_['first_name']
  769. and 'last_name' in from_
  770. and from_['last_name']
  771. ):
  772. result = result.format(
  773. name=f"{from_['first_name']} {from_['last_name']}"
  774. )
  775. elif 'first_name' in from_ and from_['first_name']:
  776. result = result.format(
  777. name=from_['first_name']
  778. )
  779. elif 'last_name' in from_ and from_['last_name']:
  780. result = result.format(
  781. name=from_['last_name']
  782. )
  783. else:
  784. result = result.format(
  785. name="Utente anonimo"
  786. )
  787. return result
  788. def datetime_from_utc_to_local(utc_datetime):
  789. """Convert `utc_datetime` to local datetime."""
  790. now_timestamp = time.time()
  791. offset = (
  792. datetime.datetime.fromtimestamp(now_timestamp)
  793. - datetime.datetime.utcfromtimestamp(now_timestamp)
  794. )
  795. return utc_datetime + offset
  796. # TIME_SYMBOLS from more specific to less specific (avoid false positives!)
  797. TIME_SYMBOLS = MyOD()
  798. TIME_SYMBOLS["'"] = 'minutes'
  799. TIME_SYMBOLS["settimana"] = 'weeks'
  800. TIME_SYMBOLS["settimane"] = 'weeks'
  801. TIME_SYMBOLS["weeks"] = 'weeks'
  802. TIME_SYMBOLS["week"] = 'weeks'
  803. TIME_SYMBOLS["giorno"] = 'days'
  804. TIME_SYMBOLS["giorni"] = 'days'
  805. TIME_SYMBOLS["secondi"] = 'seconds'
  806. TIME_SYMBOLS["seconds"] = 'seconds'
  807. TIME_SYMBOLS["secondo"] = 'seconds'
  808. TIME_SYMBOLS["minuti"] = 'minutes'
  809. TIME_SYMBOLS["minuto"] = 'minutes'
  810. TIME_SYMBOLS["minute"] = 'minutes'
  811. TIME_SYMBOLS["minutes"] = 'minutes'
  812. TIME_SYMBOLS["day"] = 'days'
  813. TIME_SYMBOLS["days"] = 'days'
  814. TIME_SYMBOLS["ore"] = 'hours'
  815. TIME_SYMBOLS["ora"] = 'hours'
  816. TIME_SYMBOLS["sec"] = 'seconds'
  817. TIME_SYMBOLS["min"] = 'minutes'
  818. TIME_SYMBOLS["m"] = 'minutes'
  819. TIME_SYMBOLS["h"] = 'hours'
  820. TIME_SYMBOLS["d"] = 'days'
  821. TIME_SYMBOLS["s"] = 'seconds'
  822. def _interval_parser(text, result):
  823. text = text.lower()
  824. succeeded = False
  825. if result is None:
  826. result = []
  827. if len(result) == 0 or result[-1]['ok']:
  828. text_part = ''
  829. _text = text # I need to iterate through _text modifying text
  830. for char in _text:
  831. if not char.isnumeric():
  832. break
  833. else:
  834. text_part += char
  835. text = text[1:]
  836. if text_part.isnumeric():
  837. result.append(
  838. dict(
  839. unit=None,
  840. value=int(text_part),
  841. ok=False
  842. )
  843. )
  844. succeeded = True, True
  845. if text:
  846. dummy, result = _interval_parser(text, result)
  847. elif len(result) > 0 and not result[-1]['ok']:
  848. text_part = ''
  849. _text = text # I need to iterate through _text modifying text
  850. for char in _text:
  851. if char.isnumeric():
  852. break
  853. else:
  854. text_part += char
  855. text = text[1:]
  856. for time_symbol, unit in TIME_SYMBOLS.items():
  857. if time_symbol in text_part:
  858. result[-1]['unit'] = unit
  859. result[-1]['ok'] = True
  860. succeeded = True, True
  861. break
  862. else:
  863. result.pop()
  864. if text:
  865. dummy, result = _interval_parser(text, result)
  866. return succeeded, result
  867. def _date_parser(text, result):
  868. succeeded = False
  869. if 3 <= len(text) <= 10 and text.count('/') >= 1:
  870. if 3 <= len(text) <= 5 and text.count('/') == 1:
  871. text += '/{:%y}'.format(datetime.datetime.now())
  872. if 6 <= len(text) <= 10 and text.count('/') == 2:
  873. day, month, year = [
  874. int(n) for n in [
  875. ''.join(char)
  876. for char in text.split('/')
  877. if char.isnumeric()
  878. ]
  879. ]
  880. if year < 100:
  881. year += 2000
  882. if result is None:
  883. result = []
  884. result += [
  885. dict(
  886. unit='day',
  887. value=day,
  888. ok=True
  889. ),
  890. dict(
  891. unit='month',
  892. value=month,
  893. ok=True
  894. ),
  895. dict(
  896. unit='year',
  897. value=year,
  898. ok=True
  899. )
  900. ]
  901. succeeded = True, True
  902. return succeeded, result
  903. def _time_parser(text, result):
  904. succeeded = False
  905. if (1 <= len(text) <= 8) and any(char.isnumeric() for char in text):
  906. text = ''.join(
  907. ':' if char == '.' else char
  908. for char in text
  909. if char.isnumeric() or char in (':', '.')
  910. )
  911. if len(text) <= 2:
  912. text = '{:02d}:00:00'.format(int(text))
  913. elif len(text) == 4 and ':' not in text:
  914. text = '{:02d}:{:02d}:00'.format(
  915. *[int(x) for x in (text[:2], text[2:])]
  916. )
  917. elif text.count(':') == 1:
  918. text = '{:02d}:{:02d}:00'.format(
  919. *[int(x) for x in text.split(':')]
  920. )
  921. if text.count(':') == 2:
  922. hour, minute, second = (int(x) for x in text.split(':'))
  923. if (
  924. 0 <= hour <= 24
  925. and 0 <= minute <= 60
  926. and 0 <= second <= 60
  927. ):
  928. if result is None:
  929. result = []
  930. result += [
  931. dict(
  932. unit='hour',
  933. value=hour,
  934. ok=True
  935. ),
  936. dict(
  937. unit='minute',
  938. value=minute,
  939. ok=True
  940. ),
  941. dict(
  942. unit='second',
  943. value=second,
  944. ok=True
  945. )
  946. ]
  947. succeeded = True
  948. return succeeded, result
  949. WEEKDAY_NAMES_ITA = ["Lunedì", "Martedì", "Mercoledì", "Giovedì",
  950. "Venerdì", "Sabato", "Domenica"]
  951. WEEKDAY_NAMES_ENG = ["Monday", "Tuesday", "Wednesday", "Thursday",
  952. "Friday", "Saturday", "Sunday"]
  953. def _period_parser(text, result):
  954. succeeded = False
  955. if text in ('every', 'ogni',):
  956. succeeded = True
  957. if text.title() in WEEKDAY_NAMES_ITA + WEEKDAY_NAMES_ENG:
  958. day_code = (WEEKDAY_NAMES_ITA + WEEKDAY_NAMES_ENG).index(text.title())
  959. if day_code > 6:
  960. day_code -= 7
  961. today = datetime.date.today()
  962. days = 1
  963. while (today + datetime.timedelta(days=days)).weekday() != day_code:
  964. days += 1
  965. if result is None:
  966. result = []
  967. result.append(
  968. dict(
  969. unit='days',
  970. value=days,
  971. ok=True,
  972. weekly=True
  973. )
  974. )
  975. succeeded = True
  976. else:
  977. succeeded, result = _interval_parser(text, result)
  978. return succeeded, result
  979. TIME_WORDS = {
  980. 'tra': dict(
  981. parser=_interval_parser,
  982. recurring=False,
  983. type_='delta'
  984. ),
  985. 'in': dict(
  986. parser=_interval_parser,
  987. recurring=False,
  988. type_='delta'
  989. ),
  990. 'at': dict(
  991. parser=_time_parser,
  992. recurring=False,
  993. type_='set'
  994. ),
  995. 'on': dict(
  996. parser=_date_parser,
  997. recurring=False,
  998. type_='set'
  999. ),
  1000. 'alle': dict(
  1001. parser=_time_parser,
  1002. recurring=False,
  1003. type_='set'
  1004. ),
  1005. 'il': dict(
  1006. parser=_date_parser,
  1007. recurring=False,
  1008. type_='set'
  1009. ),
  1010. 'every': dict(
  1011. parser=_period_parser,
  1012. recurring=True,
  1013. type_='delta'
  1014. ),
  1015. 'ogni': dict(
  1016. parser=_period_parser,
  1017. recurring=True,
  1018. type_='delta'
  1019. ),
  1020. }
  1021. def parse_datetime_interval_string(text):
  1022. """Parse `text` and return text, datetime and timedelta."""
  1023. parsers = []
  1024. result_text, result_datetime, result_timedelta = [], None, None
  1025. is_quoted_text = False
  1026. text = re.sub('\s\s+', ' ', text) # Replace multiple spaces with single space character
  1027. for word in text.split(' '):
  1028. if word.count('"') % 2:
  1029. is_quoted_text = not is_quoted_text
  1030. if is_quoted_text or '"' in word:
  1031. result_text.append(
  1032. word.replace('"', '') if 'href=' not in word else word
  1033. )
  1034. continue
  1035. result_text.append(word)
  1036. word = word.lower()
  1037. succeeded = False
  1038. if len(parsers) > 0:
  1039. succeeded, result = parsers[-1]['parser'](
  1040. word,
  1041. parsers[-1]['result']
  1042. )
  1043. if succeeded:
  1044. parsers[-1]['result'] = result
  1045. if not succeeded and word in TIME_WORDS:
  1046. parsers.append(
  1047. dict(
  1048. result=None,
  1049. parser=TIME_WORDS[word]['parser'],
  1050. recurring=TIME_WORDS[word]['recurring'],
  1051. type_=TIME_WORDS[word]['type_']
  1052. )
  1053. )
  1054. if succeeded:
  1055. result_text.pop()
  1056. if len(result_text) > 0 and result_text[-1].lower() in TIME_WORDS:
  1057. result_text.pop()
  1058. result_text = escape_html_chars(
  1059. ' '.join(result_text)
  1060. )
  1061. parsers = list(
  1062. filter(
  1063. lambda x: 'result' in x and x['result'],
  1064. parsers
  1065. )
  1066. )
  1067. recurring_event = False
  1068. weekly = False
  1069. _timedelta = datetime.timedelta()
  1070. _datetime = None
  1071. _now = datetime.datetime.now()
  1072. for parser in parsers:
  1073. if parser['recurring']:
  1074. recurring_event = True
  1075. type_ = parser['type_']
  1076. for result in parser['result']:
  1077. if not result['ok']:
  1078. continue
  1079. if recurring_event and 'weekly' in result and result['weekly']:
  1080. weekly = True
  1081. if type_ == 'set':
  1082. if _datetime is None:
  1083. _datetime = _now
  1084. _datetime = _datetime.replace(
  1085. **{
  1086. result['unit']: result['value']
  1087. }
  1088. )
  1089. elif type_ == 'delta':
  1090. _timedelta += datetime.timedelta(
  1091. **{
  1092. result['unit']: result['value']
  1093. }
  1094. )
  1095. if _datetime:
  1096. result_datetime = _datetime
  1097. if _timedelta:
  1098. if result_datetime is None:
  1099. result_datetime = _now
  1100. if recurring_event:
  1101. result_timedelta = _timedelta
  1102. if weekly:
  1103. result_timedelta = datetime.timedelta(days=7)
  1104. else:
  1105. result_datetime += _timedelta
  1106. while result_datetime and result_datetime < datetime.datetime.now():
  1107. result_datetime += (
  1108. result_timedelta
  1109. if result_timedelta
  1110. else datetime.timedelta(days=1)
  1111. )
  1112. return result_text, result_datetime, result_timedelta
  1113. DAY_GAPS = {
  1114. -1: 'ieri',
  1115. -2: 'avantieri',
  1116. 0: 'oggi',
  1117. 1: 'domani',
  1118. 2: 'dopodomani'
  1119. }
  1120. MONTH_NAMES_ITA = MyOD()
  1121. MONTH_NAMES_ITA[1] = "gennaio"
  1122. MONTH_NAMES_ITA[2] = "febbraio"
  1123. MONTH_NAMES_ITA[3] = "marzo"
  1124. MONTH_NAMES_ITA[4] = "aprile"
  1125. MONTH_NAMES_ITA[5] = "maggio"
  1126. MONTH_NAMES_ITA[6] = "giugno"
  1127. MONTH_NAMES_ITA[7] = "luglio"
  1128. MONTH_NAMES_ITA[8] = "agosto"
  1129. MONTH_NAMES_ITA[9] = "settembre"
  1130. MONTH_NAMES_ITA[10] = "ottobre"
  1131. MONTH_NAMES_ITA[11] = "novembre"
  1132. MONTH_NAMES_ITA[12] = "dicembre"
  1133. def beautytd(td):
  1134. """Format properly timedeltas."""
  1135. result = ''
  1136. if type(td) is int:
  1137. td = datetime.timedelta(seconds=td)
  1138. assert isinstance(
  1139. td,
  1140. datetime.timedelta
  1141. ), "td must be a datetime.timedelta object!"
  1142. mtd = datetime.timedelta
  1143. if td < mtd(minutes=1):
  1144. result = "{:.0f} secondi".format(
  1145. td.total_seconds()
  1146. )
  1147. elif td < mtd(minutes=60):
  1148. result = "{:.0f} min{}".format(
  1149. td.total_seconds()//60,
  1150. (
  1151. " {:.0f} s".format(
  1152. td.total_seconds() % 60
  1153. )
  1154. ) if td.total_seconds() % 60 else ''
  1155. )
  1156. elif td < mtd(days=1):
  1157. result = "{:.0f} h{}".format(
  1158. td.total_seconds()//3600,
  1159. (
  1160. " {:.0f} min".format(
  1161. (td.total_seconds() % 3600) // 60
  1162. )
  1163. ) if td.total_seconds() % 3600 else ''
  1164. )
  1165. elif td < mtd(days=30):
  1166. result = "{} giorni{}".format(
  1167. td.days,
  1168. (
  1169. " {:.0f} h".format(
  1170. td.total_seconds() % (3600*24) // 3600
  1171. )
  1172. ) if td.total_seconds() % (3600*24) else ''
  1173. )
  1174. return result
  1175. def beautydt(dt):
  1176. """Format a datetime in a smart way."""
  1177. if type(dt) is str:
  1178. dt = str_to_datetime(dt)
  1179. assert isinstance(
  1180. dt,
  1181. datetime.datetime
  1182. ), "dt must be a datetime.datetime object!"
  1183. now = datetime.datetime.now()
  1184. gap = dt - now
  1185. gap_days = (dt.date() - now.date()).days
  1186. result = "{dt:alle %H:%M}".format(
  1187. dt=dt
  1188. )
  1189. if abs(gap) < datetime.timedelta(minutes=30):
  1190. result += "{dt::%S}".format(dt=dt)
  1191. if -2 <= gap_days <= 2:
  1192. result += " di {dg}".format(
  1193. dg=DAY_GAPS[gap_days]
  1194. )
  1195. elif gap.days not in (-1, 0):
  1196. result += " del {d}{m}".format(
  1197. d=dt.day,
  1198. m=(
  1199. "" if now.year == dt.year and now.month == dt.month
  1200. else " {m}{y}".format(
  1201. m=MONTH_NAMES_ITA[dt.month].title(),
  1202. y="" if now.year == dt.year
  1203. else " {}".format(dt.year)
  1204. )
  1205. )
  1206. )
  1207. return result
  1208. HTML_SYMBOLS = MyOD()
  1209. HTML_SYMBOLS["&"] = "&amp;"
  1210. HTML_SYMBOLS["<"] = "&lt;"
  1211. HTML_SYMBOLS[">"] = "&gt;"
  1212. HTML_SYMBOLS["\""] = "&quot;"
  1213. HTML_SYMBOLS["&lt;b&gt;"] = "<b>"
  1214. HTML_SYMBOLS["&lt;/b&gt;"] = "</b>"
  1215. HTML_SYMBOLS["&lt;i&gt;"] = "<i>"
  1216. HTML_SYMBOLS["&lt;/i&gt;"] = "</i>"
  1217. HTML_SYMBOLS["&lt;code&gt;"] = "<code>"
  1218. HTML_SYMBOLS["&lt;/code&gt;"] = "</code>"
  1219. HTML_SYMBOLS["&lt;pre&gt;"] = "<pre>"
  1220. HTML_SYMBOLS["&lt;/pre&gt;"] = "</pre>"
  1221. HTML_SYMBOLS["&lt;a href=&quot;"] = "<a href=\""
  1222. HTML_SYMBOLS["&quot;&gt;"] = "\">"
  1223. HTML_SYMBOLS["&lt;/a&gt;"] = "</a>"
  1224. HTML_TAGS = [
  1225. None, "<b>", "</b>",
  1226. None, "<i>", "</i>",
  1227. None, "<code>", "</code>",
  1228. None, "<pre>", "</pre>",
  1229. None, "<a href=\"", "\">", "</a>",
  1230. None
  1231. ]
  1232. def remove_html_tags(text):
  1233. """Remove HTML tags from `text`."""
  1234. for tag in HTML_TAGS:
  1235. if tag is None:
  1236. continue
  1237. text = text.replace(tag, '')
  1238. return text
  1239. def escape_html_chars(text):
  1240. """Escape HTML chars if not part of a tag."""
  1241. for s, r in HTML_SYMBOLS.items():
  1242. text = text.replace(s, r)
  1243. copy = text
  1244. expected_tag = None
  1245. while copy:
  1246. min_ = min(
  1247. (
  1248. dict(
  1249. position=copy.find(tag) if tag in copy else len(copy),
  1250. tag=tag
  1251. )
  1252. for tag in HTML_TAGS
  1253. if tag
  1254. ),
  1255. key=lambda x: x['position'],
  1256. default=0
  1257. )
  1258. if min_['position'] == len(copy):
  1259. break
  1260. if expected_tag and min_['tag'] != expected_tag:
  1261. return text.replace('<', '_').replace('>', '_')
  1262. expected_tag = HTML_TAGS[HTML_TAGS.index(min_['tag'])+1]
  1263. copy = extract(copy, min_['tag'])
  1264. return text
  1265. def accents_to_jolly(text, lower=True):
  1266. """Replace letters with Italian accents with SQL jolly character."""
  1267. to_be_replaced = ('à', 'è', 'é', 'ì', 'ò', 'ù')
  1268. if lower:
  1269. text = text.lower()
  1270. else:
  1271. to_be_replaced += tuple(s.upper() for s in to_be_replaced)
  1272. for s in to_be_replaced:
  1273. text = text.replace(s, '_')
  1274. return text.replace("'", "''")
  1275. def get_secure_key(allowed_chars=None, length=6):
  1276. """Get a randomly-generate secure key.
  1277. You can specify a set of `allowed_chars` and a `length`.
  1278. """
  1279. if allowed_chars is None:
  1280. allowed_chars = string.ascii_uppercase + string.digits
  1281. return ''.join(
  1282. random.SystemRandom().choice(
  1283. allowed_chars
  1284. )
  1285. for _ in range(length)
  1286. )
  1287. def round_to_minute(datetime_):
  1288. """Round `datetime_` to closest minute."""
  1289. return (
  1290. datetime_ + datetime.timedelta(seconds=30)
  1291. ).replace(second=0, microsecond=0)
  1292. def get_line_by_content(text, key):
  1293. """Get line of `text` containing `key`."""
  1294. for line in text.split('\n'):
  1295. if key in line:
  1296. return line
  1297. return
  1298. def str_to_int(string):
  1299. """Cast str to int, ignoring non-numeric characters."""
  1300. string = ''.join(
  1301. char
  1302. for char in string
  1303. if char.isnumeric()
  1304. )
  1305. if len(string) == 0:
  1306. string = '0'
  1307. return int(string)
  1308. def starting_with_or_similar_to(a, b):
  1309. """Return similarity between two strings.
  1310. Least similar equals 0, most similar evaluates 1.
  1311. If similarity is less than 0.75, return 1 if one string starts with
  1312. the other and return 0.5 if one string is contained in the other.
  1313. """
  1314. a = a.lower()
  1315. b = b.lower()
  1316. similarity = SequenceMatcher(None, a, b).ratio()
  1317. if similarity < 0.75:
  1318. if b.startswith(a) or a.startswith(b):
  1319. return 1
  1320. if b in a or a in b:
  1321. return 0.5
  1322. return similarity
  1323. def pick_most_similar_from_list(list_, item):
  1324. """Return element from `list_` which is most similar to `item`.
  1325. Similarity is evaluated using `starting_with_or_similar_to`.
  1326. """
  1327. return max(
  1328. list_,
  1329. key=lambda element: starting_with_or_similar_to(
  1330. item,
  1331. element
  1332. )
  1333. )
  1334. def run_aiohttp_server(app, *args, **kwargs):
  1335. """Run an aiohttp web app, with its positional and keyword arguments.
  1336. Useful to run apps in dedicated threads.
  1337. """
  1338. loop = asyncio.new_event_loop()
  1339. asyncio.set_event_loop(loop)
  1340. web.run_app(app, *args, **kwargs)
  1341. def custom_join(_list, joiner, final=None):
  1342. """Join elements of `_list` using `joiner` (`final` as last joiner)."""
  1343. _list = list(map(str, _list))
  1344. if final is None:
  1345. final = joiner
  1346. if len(_list) == 0:
  1347. return ''
  1348. if len(_list) == 1:
  1349. return _list[0]
  1350. if len(_list) == 2:
  1351. return final.join(_list)
  1352. return joiner.join(_list[:-1]) + final + _list[-1]
  1353. def make_inline_query_answer(answer):
  1354. """Return an article-type answer to inline query.
  1355. Takes either a string or a dictionary and returns a list.
  1356. """
  1357. if type(answer) is str:
  1358. answer = dict(
  1359. type='article',
  1360. id=0,
  1361. title=remove_html_tags(answer),
  1362. input_message_content=dict(
  1363. message_text=answer,
  1364. parse_mode='HTML'
  1365. )
  1366. )
  1367. if type(answer) is dict:
  1368. answer = [answer]
  1369. return answer
  1370. async def dummy_coroutine(*args, **kwargs):
  1371. """Accept everthing as argument and do nothing."""
  1372. return
  1373. async def send_csv_file(bot, chat_id, query, caption=None,
  1374. file_name='File.csv', user_record=None, update=dict()):
  1375. """Run a query on `bot` database and send result as CSV file to `chat_id`.
  1376. Optional parameters `caption` and `file_name` may be passed to this
  1377. function.
  1378. """
  1379. try:
  1380. with bot.db as db:
  1381. record = db.query(
  1382. query
  1383. )
  1384. header_line = []
  1385. body_lines = []
  1386. for row in record:
  1387. if not header_line:
  1388. header_line.append(get_csv_string(row.keys()))
  1389. body_lines.append(get_csv_string(row.values()))
  1390. text = '\n'.join(header_line + body_lines)
  1391. except Exception as e:
  1392. text = "{message}\n{e}".format(
  1393. message=bot.get_message('admin', 'query_button', 'error',
  1394. user_record=user_record, update=update),
  1395. e=e
  1396. )
  1397. for x, y in {'&lt;': '<', '\n': '\r\n'}.items():
  1398. text = text.replace(x, y)
  1399. if len(text) == 0:
  1400. text = bot.get_message('admin', 'query_button', 'empty_file',
  1401. user_record=user_record, update=update)
  1402. with io.BytesIO(text.encode('utf-8')) as f:
  1403. f.name = file_name
  1404. return await bot.send_document(
  1405. chat_id=chat_id,
  1406. document=f,
  1407. caption=caption
  1408. )
  1409. async def send_part_of_text_file(bot, chat_id, file_path, caption=None,
  1410. file_name='File.txt', user_record=None,
  1411. update=dict(),
  1412. reversed_=True,
  1413. limit=None):
  1414. """Send `lines` lines of text file via `bot` in `chat_id`.
  1415. If `reversed`, read the file from last line.
  1416. TODO: do not load whole file in RAM. At the moment this is the easiest
  1417. way to allow `reversed` files, but it is inefficient and requires a lot
  1418. of memory.
  1419. """
  1420. try:
  1421. with open(file_path, 'r') as log_file:
  1422. lines = log_file.readlines()
  1423. if reversed:
  1424. lines = lines[::-1]
  1425. if limit:
  1426. lines = lines[:limit]
  1427. with io.BytesIO(
  1428. ''.join(lines).encode('utf-8')
  1429. ) as document:
  1430. document.name = file_name
  1431. return await bot.send_document(
  1432. chat_id=chat_id,
  1433. document=document,
  1434. caption=caption
  1435. )
  1436. except Exception as e:
  1437. return e