mirror of
https://github.com/LonamiWebs/Telethon.git
synced 2024-11-25 19:03:46 +03:00
165 lines
5.6 KiB
Python
165 lines
5.6 KiB
Python
import inspect
|
|
|
|
from .tl import types
|
|
|
|
|
|
# Which updates have the following fields?
|
|
_has_channel_id = []
|
|
|
|
|
|
# TODO EntityCache does the same. Reuse?
|
|
def _fill():
|
|
for name in dir(types):
|
|
update = getattr(types, name)
|
|
if getattr(update, 'SUBCLASS_OF_ID', None) == 0x9f89304e:
|
|
cid = update.CONSTRUCTOR_ID
|
|
sig = inspect.signature(update.__init__)
|
|
for param in sig.parameters.values():
|
|
if param.name == 'channel_id' and param.annotation == int:
|
|
_has_channel_id.append(cid)
|
|
|
|
if not _has_channel_id:
|
|
raise RuntimeError('FIXME: Did the init signature or updates change?')
|
|
|
|
|
|
# We use a function to avoid cluttering the globals (with name/update/cid/doc)
|
|
_fill()
|
|
|
|
|
|
class StateCache:
|
|
"""
|
|
In-memory update state cache, defaultdict-like behaviour.
|
|
"""
|
|
def __init__(self, initial, loggers):
|
|
# We only care about the pts and the date. By using a tuple which
|
|
# is lightweight and immutable we can easily copy them around to
|
|
# each update in case they need to fetch missing entities.
|
|
self._logger = loggers[__name__]
|
|
if initial:
|
|
self._pts_date = initial.pts, initial.date
|
|
else:
|
|
self._pts_date = None, None
|
|
|
|
def reset(self):
|
|
self.__dict__.clear()
|
|
self._pts_date = None, None
|
|
|
|
# TODO Call this when receiving responses too...?
|
|
def update(
|
|
self,
|
|
update,
|
|
*,
|
|
channel_id=None,
|
|
has_pts=frozenset(x.CONSTRUCTOR_ID for x in (
|
|
types.UpdateNewMessage,
|
|
types.UpdateDeleteMessages,
|
|
types.UpdateReadHistoryInbox,
|
|
types.UpdateReadHistoryOutbox,
|
|
types.UpdateWebPage,
|
|
types.UpdateReadMessagesContents,
|
|
types.UpdateEditMessage,
|
|
types.updates.State,
|
|
types.updates.DifferenceTooLong,
|
|
types.UpdateShortMessage,
|
|
types.UpdateShortChatMessage,
|
|
types.UpdateShortSentMessage
|
|
)),
|
|
has_date=frozenset(x.CONSTRUCTOR_ID for x in (
|
|
types.UpdateUserPhoto,
|
|
types.UpdateEncryption,
|
|
types.UpdateEncryptedMessagesRead,
|
|
types.UpdateChatParticipantAdd,
|
|
types.updates.DifferenceEmpty,
|
|
types.UpdateShortMessage,
|
|
types.UpdateShortChatMessage,
|
|
types.UpdateShort,
|
|
types.UpdatesCombined,
|
|
types.Updates,
|
|
types.UpdateShortSentMessage,
|
|
)),
|
|
has_channel_pts=frozenset(x.CONSTRUCTOR_ID for x in (
|
|
types.UpdateChannelTooLong,
|
|
types.UpdateNewChannelMessage,
|
|
types.UpdateDeleteChannelMessages,
|
|
types.UpdateEditChannelMessage,
|
|
types.UpdateChannelWebPage,
|
|
types.updates.ChannelDifferenceEmpty,
|
|
types.updates.ChannelDifferenceTooLong,
|
|
types.updates.ChannelDifference
|
|
)),
|
|
check_only=False
|
|
):
|
|
"""
|
|
Update the state with the given update.
|
|
"""
|
|
cid = update.CONSTRUCTOR_ID
|
|
if check_only:
|
|
return cid in has_pts or cid in has_date or cid in has_channel_pts
|
|
|
|
if cid in has_pts:
|
|
if cid in has_date:
|
|
self._pts_date = update.pts, update.date
|
|
else:
|
|
self._pts_date = update.pts, self._pts_date[1]
|
|
elif cid in has_date:
|
|
self._pts_date = self._pts_date[0], update.date
|
|
|
|
if cid in has_channel_pts:
|
|
if channel_id is None:
|
|
channel_id = self.get_channel_id(update)
|
|
|
|
if channel_id is None:
|
|
self._logger.info(
|
|
'Failed to retrieve channel_id from %s', update)
|
|
else:
|
|
self.__dict__[channel_id] = update.pts
|
|
|
|
def get_channel_id(
|
|
self,
|
|
update,
|
|
has_channel_id=frozenset(_has_channel_id),
|
|
# Hardcoded because only some with message are for channels
|
|
has_message=frozenset(x.CONSTRUCTOR_ID for x in (
|
|
types.UpdateNewChannelMessage,
|
|
types.UpdateEditChannelMessage
|
|
))
|
|
):
|
|
"""
|
|
Gets the **unmarked** channel ID from this update, if it has any.
|
|
|
|
Fails for ``*difference`` updates, where ``channel_id``
|
|
is supposedly already known from the outside.
|
|
"""
|
|
cid = update.CONSTRUCTOR_ID
|
|
if cid in has_channel_id:
|
|
return update.channel_id
|
|
elif cid in has_message:
|
|
if update.message.peer_id is None:
|
|
# Telegram sometimes sends empty messages to give a newer pts:
|
|
# UpdateNewChannelMessage(message=MessageEmpty(id), pts=pts, pts_count=1)
|
|
# Not sure why, but it's safe to ignore them.
|
|
self._logger.debug('Update has None peer_id %s', update)
|
|
else:
|
|
return update.message.peer_id.channel_id
|
|
|
|
return None
|
|
|
|
def __getitem__(self, item):
|
|
"""
|
|
If `item` is `None`, returns the default ``(pts, date)``.
|
|
|
|
If it's an **unmarked** channel ID, returns its ``pts``.
|
|
|
|
If no information is known, ``pts`` will be `None`.
|
|
"""
|
|
if item is None:
|
|
return self._pts_date
|
|
else:
|
|
return self.__dict__.get(item)
|
|
|
|
def __setitem__(self, where, value):
|
|
if where is None:
|
|
self._pts_date = value
|
|
else:
|
|
self.__dict__[where] = value
|