2018-03-02 00:34:32 +03:00
|
|
|
from enum import Enum
|
|
|
|
|
|
|
|
from .abstract import Session
|
2018-03-08 12:05:40 +03:00
|
|
|
from .. import utils
|
2018-03-02 00:34:32 +03:00
|
|
|
from ..tl import TLObject
|
|
|
|
from ..tl.types import (
|
|
|
|
PeerUser, PeerChat, PeerChannel,
|
|
|
|
InputPeerUser, InputPeerChat, InputPeerChannel,
|
|
|
|
InputPhoto, InputDocument
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
class _SentFileType(Enum):
|
|
|
|
DOCUMENT = 0
|
|
|
|
PHOTO = 1
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def from_type(cls):
|
|
|
|
if cls == InputDocument:
|
|
|
|
return _SentFileType.DOCUMENT
|
|
|
|
elif cls == InputPhoto:
|
|
|
|
return _SentFileType.PHOTO
|
|
|
|
else:
|
|
|
|
raise ValueError('The cls must be either InputDocument/InputPhoto')
|
|
|
|
|
|
|
|
|
|
|
|
class MemorySession(Session):
|
|
|
|
def __init__(self):
|
2018-03-02 12:10:11 +03:00
|
|
|
super().__init__()
|
2018-03-02 19:39:04 +03:00
|
|
|
|
2018-03-03 16:59:30 +03:00
|
|
|
self._dc_id = 0
|
2018-03-02 00:34:32 +03:00
|
|
|
self._server_address = None
|
|
|
|
self._port = None
|
|
|
|
self._auth_key = None
|
2019-02-10 13:10:41 +03:00
|
|
|
self._takeout_id = None
|
2018-03-02 00:34:32 +03:00
|
|
|
|
|
|
|
self._files = {}
|
|
|
|
self._entities = set()
|
2018-04-25 14:37:29 +03:00
|
|
|
self._update_states = {}
|
2018-03-02 00:34:32 +03:00
|
|
|
|
|
|
|
def set_dc(self, dc_id, server_address, port):
|
2018-03-03 16:59:30 +03:00
|
|
|
self._dc_id = dc_id or 0
|
2018-03-02 00:34:32 +03:00
|
|
|
self._server_address = server_address
|
|
|
|
self._port = port
|
|
|
|
|
2018-07-21 12:59:44 +03:00
|
|
|
@property
|
|
|
|
def dc_id(self):
|
|
|
|
return self._dc_id
|
|
|
|
|
2018-03-02 00:34:32 +03:00
|
|
|
@property
|
|
|
|
def server_address(self):
|
|
|
|
return self._server_address
|
|
|
|
|
|
|
|
@property
|
|
|
|
def port(self):
|
|
|
|
return self._port
|
|
|
|
|
|
|
|
@property
|
|
|
|
def auth_key(self):
|
|
|
|
return self._auth_key
|
|
|
|
|
|
|
|
@auth_key.setter
|
|
|
|
def auth_key(self, value):
|
|
|
|
self._auth_key = value
|
|
|
|
|
2019-02-10 13:10:41 +03:00
|
|
|
@property
|
|
|
|
def takeout_id(self):
|
|
|
|
return self._takeout_id
|
|
|
|
|
|
|
|
@takeout_id.setter
|
|
|
|
def takeout_id(self, value):
|
|
|
|
self._takeout_id = value
|
|
|
|
|
2018-04-25 14:37:29 +03:00
|
|
|
def get_update_state(self, entity_id):
|
|
|
|
return self._update_states.get(entity_id, None)
|
|
|
|
|
|
|
|
def set_update_state(self, entity_id, state):
|
|
|
|
self._update_states[entity_id] = state
|
|
|
|
|
2018-03-02 00:34:32 +03:00
|
|
|
def close(self):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def save(self):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def delete(self):
|
|
|
|
pass
|
|
|
|
|
2019-05-03 14:59:17 +03:00
|
|
|
@staticmethod
|
|
|
|
def _entity_values_to_row(id, hash, username, phone, name):
|
2018-03-03 18:48:57 +03:00
|
|
|
# While this is a simple implementation it might be overrode by,
|
|
|
|
# other classes so they don't need to implement the plural form
|
|
|
|
# of the method. Don't remove.
|
|
|
|
return id, hash, username, phone, name
|
|
|
|
|
2018-03-02 14:20:11 +03:00
|
|
|
def _entity_to_row(self, e):
|
|
|
|
if not isinstance(e, TLObject):
|
|
|
|
return
|
|
|
|
try:
|
|
|
|
p = utils.get_input_peer(e, allow_self=False)
|
|
|
|
marked_id = utils.get_peer_id(p)
|
2018-03-24 14:09:33 +03:00
|
|
|
except TypeError:
|
2019-09-12 20:17:32 +03:00
|
|
|
# Note: `get_input_peer` already checks for non-zero `access_hash`.
|
|
|
|
# See issues #354 and #392. It also checks that the entity
|
|
|
|
# is not `min`, because its `access_hash` cannot be used
|
|
|
|
# anywhere (since layer 102, there are two access hashes).
|
2018-03-02 14:20:11 +03:00
|
|
|
return
|
|
|
|
|
|
|
|
if isinstance(p, (InputPeerUser, InputPeerChannel)):
|
2019-03-26 13:27:21 +03:00
|
|
|
p_hash = p.access_hash
|
2018-03-02 14:20:11 +03:00
|
|
|
elif isinstance(p, InputPeerChat):
|
|
|
|
p_hash = 0
|
|
|
|
else:
|
|
|
|
return
|
|
|
|
|
|
|
|
username = getattr(e, 'username', None) or None
|
|
|
|
if username is not None:
|
|
|
|
username = username.lower()
|
|
|
|
phone = getattr(e, 'phone', None)
|
|
|
|
name = utils.get_display_name(e) or None
|
2018-03-03 18:48:57 +03:00
|
|
|
return self._entity_values_to_row(
|
|
|
|
marked_id, p_hash, username, phone, name
|
|
|
|
)
|
2018-03-02 14:20:11 +03:00
|
|
|
|
|
|
|
def _entities_to_rows(self, tlo):
|
2018-03-02 00:34:32 +03:00
|
|
|
if not isinstance(tlo, TLObject) and utils.is_list_like(tlo):
|
|
|
|
# This may be a list of users already for instance
|
|
|
|
entities = tlo
|
|
|
|
else:
|
|
|
|
entities = []
|
2018-08-02 20:40:48 +03:00
|
|
|
if hasattr(tlo, 'user'):
|
|
|
|
entities.append(tlo.user)
|
2018-03-02 00:34:32 +03:00
|
|
|
if hasattr(tlo, 'chats') and utils.is_list_like(tlo.chats):
|
|
|
|
entities.extend(tlo.chats)
|
|
|
|
if hasattr(tlo, 'users') and utils.is_list_like(tlo.users):
|
|
|
|
entities.extend(tlo.users)
|
|
|
|
|
|
|
|
rows = [] # Rows to add (id, hash, username, phone, name)
|
|
|
|
for e in entities:
|
2018-03-02 14:20:11 +03:00
|
|
|
row = self._entity_to_row(e)
|
|
|
|
if row:
|
|
|
|
rows.append(row)
|
2018-03-02 00:34:32 +03:00
|
|
|
return rows
|
|
|
|
|
|
|
|
def process_entities(self, tlo):
|
2018-03-03 19:08:49 +03:00
|
|
|
self._entities |= set(self._entities_to_rows(tlo))
|
2018-03-02 00:34:32 +03:00
|
|
|
|
|
|
|
def get_entity_rows_by_phone(self, phone):
|
2018-03-03 14:51:35 +03:00
|
|
|
try:
|
|
|
|
return next((id, hash) for id, hash, _, found_phone, _
|
|
|
|
in self._entities if found_phone == phone)
|
|
|
|
except StopIteration:
|
|
|
|
pass
|
2018-03-02 00:34:32 +03:00
|
|
|
|
|
|
|
def get_entity_rows_by_username(self, username):
|
2018-03-03 14:51:35 +03:00
|
|
|
try:
|
|
|
|
return next((id, hash) for id, hash, found_username, _, _
|
|
|
|
in self._entities if found_username == username)
|
|
|
|
except StopIteration:
|
|
|
|
pass
|
2018-03-02 00:34:32 +03:00
|
|
|
|
|
|
|
def get_entity_rows_by_name(self, name):
|
2018-03-03 14:51:35 +03:00
|
|
|
try:
|
|
|
|
return next((id, hash) for id, hash, _, _, found_name
|
|
|
|
in self._entities if found_name == name)
|
|
|
|
except StopIteration:
|
|
|
|
pass
|
2018-03-02 00:34:32 +03:00
|
|
|
|
2018-03-08 12:05:40 +03:00
|
|
|
def get_entity_rows_by_id(self, id, exact=True):
|
2018-03-03 14:51:35 +03:00
|
|
|
try:
|
2018-03-08 12:05:40 +03:00
|
|
|
if exact:
|
|
|
|
return next((id, hash) for found_id, hash, _, _, _
|
|
|
|
in self._entities if found_id == id)
|
|
|
|
else:
|
|
|
|
ids = (
|
|
|
|
utils.get_peer_id(PeerUser(id)),
|
|
|
|
utils.get_peer_id(PeerChat(id)),
|
|
|
|
utils.get_peer_id(PeerChannel(id))
|
|
|
|
)
|
|
|
|
return next((id, hash) for found_id, hash, _, _, _
|
|
|
|
in self._entities if found_id in ids)
|
2018-03-03 14:51:35 +03:00
|
|
|
except StopIteration:
|
|
|
|
pass
|
2018-03-02 00:34:32 +03:00
|
|
|
|
|
|
|
def get_input_entity(self, key):
|
|
|
|
try:
|
|
|
|
if key.SUBCLASS_OF_ID in (0xc91c90b6, 0xe669bf46, 0x40f202fd):
|
|
|
|
# hex(crc32(b'InputPeer', b'InputUser' and b'InputChannel'))
|
|
|
|
# We already have an Input version, so nothing else required
|
|
|
|
return key
|
|
|
|
# Try to early return if this key can be casted as input peer
|
|
|
|
return utils.get_input_peer(key)
|
|
|
|
except (AttributeError, TypeError):
|
|
|
|
# Not a TLObject or can't be cast into InputPeer
|
|
|
|
if isinstance(key, TLObject):
|
|
|
|
key = utils.get_peer_id(key)
|
2018-03-08 12:05:40 +03:00
|
|
|
exact = True
|
|
|
|
else:
|
2018-03-08 12:12:43 +03:00
|
|
|
exact = not isinstance(key, int) or key < 0
|
2018-03-02 00:34:32 +03:00
|
|
|
|
|
|
|
result = None
|
|
|
|
if isinstance(key, str):
|
|
|
|
phone = utils.parse_phone(key)
|
|
|
|
if phone:
|
|
|
|
result = self.get_entity_rows_by_phone(phone)
|
|
|
|
else:
|
2018-08-02 14:47:35 +03:00
|
|
|
username, invite = utils.parse_username(key)
|
|
|
|
if username and not invite:
|
2018-03-02 00:34:32 +03:00
|
|
|
result = self.get_entity_rows_by_username(username)
|
2018-08-02 14:47:35 +03:00
|
|
|
else:
|
2018-11-15 19:23:47 +03:00
|
|
|
tup = utils.resolve_invite_link(key)[1]
|
|
|
|
if tup:
|
|
|
|
result = self.get_entity_rows_by_id(tup, exact=False)
|
2018-03-02 00:34:32 +03:00
|
|
|
|
2018-03-08 12:05:40 +03:00
|
|
|
elif isinstance(key, int):
|
|
|
|
result = self.get_entity_rows_by_id(key, exact)
|
2018-03-02 00:34:32 +03:00
|
|
|
|
|
|
|
if not result and isinstance(key, str):
|
|
|
|
result = self.get_entity_rows_by_name(key)
|
|
|
|
|
|
|
|
if result:
|
2018-04-03 14:05:01 +03:00
|
|
|
entity_id, entity_hash = result # unpack resulting tuple
|
|
|
|
entity_id, kind = utils.resolve_id(entity_id)
|
|
|
|
# removes the mark and returns type of entity
|
|
|
|
if kind == PeerUser:
|
|
|
|
return InputPeerUser(entity_id, entity_hash)
|
|
|
|
elif kind == PeerChat:
|
|
|
|
return InputPeerChat(entity_id)
|
|
|
|
elif kind == PeerChannel:
|
|
|
|
return InputPeerChannel(entity_id, entity_hash)
|
2018-03-02 00:34:32 +03:00
|
|
|
else:
|
|
|
|
raise ValueError('Could not find input entity with key ', key)
|
|
|
|
|
|
|
|
def cache_file(self, md5_digest, file_size, instance):
|
|
|
|
if not isinstance(instance, (InputDocument, InputPhoto)):
|
|
|
|
raise TypeError('Cannot cache %s instance' % type(instance))
|
2018-08-22 17:21:22 +03:00
|
|
|
key = (md5_digest, file_size, _SentFileType.from_type(type(instance)))
|
2018-03-02 00:34:32 +03:00
|
|
|
value = (instance.id, instance.access_hash)
|
|
|
|
self._files[key] = value
|
|
|
|
|
|
|
|
def get_file(self, md5_digest, file_size, cls):
|
|
|
|
key = (md5_digest, file_size, _SentFileType.from_type(cls))
|
|
|
|
try:
|
2018-11-01 12:29:23 +03:00
|
|
|
return cls(*self._files[key])
|
2018-03-02 00:34:32 +03:00
|
|
|
except KeyError:
|
|
|
|
return None
|