2016-10-25 19:28:34 +02:00
|
|
|
#!/usr/bin/env python
|
|
|
|
#
|
|
|
|
# A library that provides a Python interface to the Telegram Bot API
|
2018-01-04 16:16:06 +01:00
|
|
|
# Copyright (C) 2015-2018
|
2016-10-25 19:28:34 +02:00
|
|
|
# Leandro Toledo de Souza <devs@python-telegram-bot.org>
|
|
|
|
#
|
|
|
|
# This program is free software: you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU Lesser Public License as published by
|
|
|
|
# the Free Software Foundation, either version 3 of the License, or
|
|
|
|
# (at your option) any later version.
|
|
|
|
#
|
|
|
|
# This program is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
# GNU Lesser Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU Lesser Public License
|
|
|
|
# along with this program. If not, see [http://www.gnu.org/licenses/].
|
2017-09-01 08:43:08 +02:00
|
|
|
"""This module contains helper functions."""
|
2018-09-20 22:50:40 +02:00
|
|
|
from collections import defaultdict
|
|
|
|
|
|
|
|
try:
|
|
|
|
import ujson as json
|
|
|
|
except ImportError:
|
|
|
|
import json
|
2018-02-22 16:38:54 +01:00
|
|
|
from html import escape
|
2016-10-25 19:28:34 +02:00
|
|
|
|
2017-01-20 20:13:58 +01:00
|
|
|
import re
|
2018-01-09 16:54:07 +01:00
|
|
|
import signal
|
2017-07-01 17:08:45 +02:00
|
|
|
from datetime import datetime
|
2016-10-25 19:28:34 +02:00
|
|
|
|
2018-01-09 16:54:07 +01:00
|
|
|
# From https://stackoverflow.com/questions/2549939/get-signal-names-from-numbers-in-python
|
|
|
|
_signames = {v: k
|
|
|
|
for k, v in reversed(sorted(vars(signal).items()))
|
|
|
|
if k.startswith('SIG') and not k.startswith('SIG_')}
|
|
|
|
|
|
|
|
|
|
|
|
def get_signal_name(signum):
|
|
|
|
"""Returns the signal name of the given signal number."""
|
|
|
|
return _signames[signum]
|
|
|
|
|
|
|
|
|
2017-07-01 17:08:45 +02:00
|
|
|
# Not using future.backports.datetime here as datetime value might be an input from the user,
|
|
|
|
# making every isinstace() call more delicate. So we just use our own compat layer.
|
|
|
|
if hasattr(datetime, 'timestamp'):
|
|
|
|
# Python 3.3+
|
|
|
|
def _timestamp(dt_obj):
|
|
|
|
return dt_obj.timestamp()
|
|
|
|
else:
|
|
|
|
# Python < 3.3 (incl 2.7)
|
|
|
|
from time import mktime
|
|
|
|
|
|
|
|
def _timestamp(dt_obj):
|
|
|
|
return mktime(dt_obj.timetuple())
|
|
|
|
|
2016-10-25 19:28:34 +02:00
|
|
|
|
2017-01-20 20:13:58 +01:00
|
|
|
def escape_markdown(text):
|
2017-09-01 08:43:08 +02:00
|
|
|
"""Helper function to escape telegram markup symbols."""
|
2017-01-20 20:13:58 +01:00
|
|
|
escape_chars = '\*_`\['
|
|
|
|
return re.sub(r'([%s])' % escape_chars, r'\\\1', text)
|
2017-07-01 17:08:45 +02:00
|
|
|
|
|
|
|
|
|
|
|
def to_timestamp(dt_obj):
|
|
|
|
"""
|
|
|
|
Args:
|
|
|
|
dt_obj (:class:`datetime.datetime`):
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
int:
|
2017-09-01 08:43:08 +02:00
|
|
|
|
2017-07-01 17:08:45 +02:00
|
|
|
"""
|
|
|
|
if not dt_obj:
|
|
|
|
return None
|
|
|
|
|
|
|
|
return int(_timestamp(dt_obj))
|
|
|
|
|
|
|
|
|
|
|
|
def from_timestamp(unixtime):
|
|
|
|
"""
|
|
|
|
Args:
|
|
|
|
unixtime (int):
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
datetime.datetime:
|
2017-09-01 08:43:08 +02:00
|
|
|
|
2017-07-01 17:08:45 +02:00
|
|
|
"""
|
|
|
|
if not unixtime:
|
|
|
|
return None
|
|
|
|
|
|
|
|
return datetime.fromtimestamp(unixtime)
|
2017-09-01 08:40:05 +02:00
|
|
|
|
|
|
|
|
|
|
|
def mention_html(user_id, name):
|
|
|
|
"""
|
|
|
|
Args:
|
|
|
|
user_id (:obj:`int`) The user's id which you want to mention.
|
|
|
|
name (:obj:`str`) The name the mention is showing.
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
:obj:`str`: The inline mention for the user as html.
|
|
|
|
"""
|
|
|
|
if isinstance(user_id, int):
|
2018-05-28 22:51:39 +02:00
|
|
|
return u'<a href="tg://user?id={}">{}</a>'.format(user_id, escape(name))
|
2017-09-01 08:40:05 +02:00
|
|
|
|
|
|
|
|
|
|
|
def mention_markdown(user_id, name):
|
|
|
|
"""
|
|
|
|
Args:
|
|
|
|
user_id (:obj:`int`) The user's id which you want to mention.
|
|
|
|
name (:obj:`str`) The name the mention is showing.
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
:obj:`str`: The inline mention for the user as markdown.
|
|
|
|
"""
|
|
|
|
if isinstance(user_id, int):
|
2018-05-28 22:51:39 +02:00
|
|
|
return u'[{}](tg://user?id={})'.format(escape_markdown(name), user_id)
|
2018-02-15 10:21:19 +01:00
|
|
|
|
|
|
|
|
|
|
|
def effective_message_type(entity):
|
|
|
|
"""
|
|
|
|
Extracts the type of message as a string identifier from a :class:`telegram.Message` or a
|
|
|
|
:class:`telegram.Update`.
|
|
|
|
|
|
|
|
Args:
|
|
|
|
entity (:obj:`Update` | :obj:`Message`) The ``update`` or ``message`` to extract from
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
str: One of ``Message.MESSAGE_TYPES``
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
# Importing on file-level yields cyclic Import Errors
|
|
|
|
from telegram import Message
|
|
|
|
from telegram import Update
|
|
|
|
|
|
|
|
if isinstance(entity, Message):
|
|
|
|
message = entity
|
|
|
|
elif isinstance(entity, Update):
|
|
|
|
message = entity.effective_message
|
|
|
|
else:
|
|
|
|
raise TypeError("entity is not Message or Update (got: {})".format(type(entity)))
|
|
|
|
|
|
|
|
for i in Message.MESSAGE_TYPES:
|
|
|
|
if getattr(message, i, None):
|
|
|
|
return i
|
|
|
|
|
|
|
|
return None
|
2018-09-20 22:50:40 +02:00
|
|
|
|
|
|
|
|
|
|
|
def enocde_conversations_to_json(conversations):
|
|
|
|
"""Helper method to encode a conversations dict (that uses tuples as keys) to a
|
|
|
|
JSON-serializable way. Use :attr:`_decode_conversations_from_json` to decode.
|
|
|
|
|
|
|
|
Args:
|
|
|
|
conversations (:obj:`dict`): The conversations dict to transofrm to JSON.
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
:obj:`str`: The JSON-serialized conversations dict
|
|
|
|
"""
|
|
|
|
tmp = {}
|
|
|
|
for handler, states in conversations.items():
|
|
|
|
tmp[handler] = {}
|
|
|
|
for key, state in states.items():
|
|
|
|
tmp[handler][json.dumps(key)] = state
|
|
|
|
return json.dumps(tmp)
|
|
|
|
|
|
|
|
|
|
|
|
def decode_conversations_from_json(json_string):
|
|
|
|
"""Helper method to decode a conversations dict (that uses tuples as keys) from a
|
|
|
|
JSON-string created with :attr:`_encode_conversations_to_json`.
|
|
|
|
|
|
|
|
Args:
|
|
|
|
json_string (:obj:`str`): The conversations dict as JSON string.
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
:obj:`dict`: The conversations dict after decoding
|
|
|
|
"""
|
|
|
|
tmp = json.loads(json_string)
|
|
|
|
conversations = {}
|
|
|
|
for handler, states in tmp.items():
|
|
|
|
conversations[handler] = {}
|
|
|
|
for key, state in states.items():
|
|
|
|
conversations[handler][tuple(json.loads(key))] = state
|
|
|
|
return conversations
|
|
|
|
|
|
|
|
|
|
|
|
def decode_user_chat_data_from_json(data):
|
|
|
|
"""Helper method to decode chat or user data (that uses ints as keys) from a
|
|
|
|
JSON-string.
|
|
|
|
|
|
|
|
Args:
|
|
|
|
data (:obj:`str`): The user/chat_data dict as JSON string.
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
:obj:`dict`: The user/chat_data defaultdict after decoding
|
|
|
|
"""
|
|
|
|
|
|
|
|
tmp = defaultdict(dict)
|
|
|
|
decoded_data = json.loads(data)
|
|
|
|
for user, data in decoded_data.items():
|
|
|
|
user = int(user)
|
|
|
|
tmp[user] = {}
|
|
|
|
for key, value in data.items():
|
|
|
|
try:
|
|
|
|
key = int(key)
|
|
|
|
except ValueError:
|
|
|
|
pass
|
|
|
|
tmp[user][key] = value
|
|
|
|
return tmp
|