mirror of
https://github.com/LonamiWebs/Telethon.git
synced 2024-11-26 03:13:45 +03:00
f8264abb5a
Entity cache uses are removed. It was a source of ever-growing memory usage that has to be reworked. This affects everything that tried to obtain an input entity, input sender or input chat (such as the SenderGetter or calls to _get_entity_pair). Input entities need to be reworked in any case. Its removal also affects the automatic cache of any raw API request. Raise last error parameter is removed, and its behaviour made default. The connection type parameter has been removed, since users really have no need to change it. A few more attributes have been made private, since users should not mess with those.
180 lines
5.8 KiB
Python
180 lines
5.8 KiB
Python
import abc
|
|
import asyncio
|
|
import warnings
|
|
|
|
from .. import _tl
|
|
from .._misc import utils, tlobject
|
|
from ..types._custom.chatgetter import ChatGetter
|
|
|
|
|
|
async def _into_id_set(client, chats):
|
|
"""Helper util to turn the input chat or chats into a set of IDs."""
|
|
if chats is None:
|
|
return None
|
|
|
|
if not utils.is_list_like(chats):
|
|
chats = (chats,)
|
|
|
|
result = set()
|
|
for chat in chats:
|
|
if isinstance(chat, int):
|
|
result.add(chat)
|
|
elif isinstance(chat, tlobject.TLObject) and chat.SUBCLASS_OF_ID == 0x2d45687:
|
|
# 0x2d45687 == crc32(b'Peer')
|
|
result.add(utils.get_peer_id(chat))
|
|
else:
|
|
chat = await client.get_input_entity(chat)
|
|
if isinstance(chat, _tl.InputPeerSelf):
|
|
chat = await client.get_me(input_peer=True)
|
|
result.add(utils.get_peer_id(chat))
|
|
|
|
return result
|
|
|
|
|
|
class EventBuilder(abc.ABC):
|
|
"""
|
|
The common event builder, with builtin support to filter per chat.
|
|
|
|
Args:
|
|
chats (`entity`, optional):
|
|
May be one or more entities (username/peer/etc.), preferably IDs.
|
|
By default, only matching chats will be handled.
|
|
|
|
blacklist_chats (`bool`, optional):
|
|
Whether to treat the chats as a blacklist instead of
|
|
as a whitelist (default). This means that every chat
|
|
will be handled *except* those specified in ``chats``
|
|
which will be ignored if ``blacklist_chats=True``.
|
|
|
|
func (`callable`, optional):
|
|
A callable (async or not) function that should accept the event as input
|
|
parameter, and return a value indicating whether the event
|
|
should be dispatched or not (any truthy value will do, it
|
|
does not need to be a `bool`). It works like a custom filter:
|
|
|
|
.. code-block:: python
|
|
|
|
@client.on(events.NewMessage(func=lambda e: e.is_private))
|
|
async def handler(event):
|
|
pass # code here
|
|
"""
|
|
def __init__(self, chats=None, *, blacklist_chats=False, func=None):
|
|
self.chats = chats
|
|
self.blacklist_chats = bool(blacklist_chats)
|
|
self.resolved = False
|
|
self.func = func
|
|
self._resolve_lock = None
|
|
|
|
@classmethod
|
|
@abc.abstractmethod
|
|
def build(cls, update, others, self_id, entities, client):
|
|
"""
|
|
Builds an event for the given update if possible, or returns None.
|
|
|
|
`others` are the rest of updates that came in the same container
|
|
as the current `update`.
|
|
|
|
`self_id` should be the current user's ID, since it is required
|
|
for some events which lack this information but still need it.
|
|
"""
|
|
# TODO So many parameters specific to only some update types seems dirty
|
|
|
|
async def resolve(self, client):
|
|
"""Helper method to allow event builders to be resolved before usage"""
|
|
if self.resolved:
|
|
return
|
|
|
|
if not self._resolve_lock:
|
|
self._resolve_lock = asyncio.Lock()
|
|
|
|
async with self._resolve_lock:
|
|
if not self.resolved:
|
|
await self._resolve(client)
|
|
self.resolved = True
|
|
|
|
async def _resolve(self, client):
|
|
self.chats = await _into_id_set(client, self.chats)
|
|
|
|
def filter(self, event):
|
|
"""
|
|
Returns a truthy value if the event passed the filter and should be
|
|
used, or falsy otherwise. The return value may need to be awaited.
|
|
|
|
The events must have been resolved before this can be called.
|
|
"""
|
|
if not self.resolved:
|
|
return
|
|
|
|
if self.chats is not None:
|
|
# Note: the `event.chat_id` property checks if it's `None` for us
|
|
inside = event.chat_id in self.chats
|
|
if inside == self.blacklist_chats:
|
|
# If this chat matches but it's a blacklist ignore.
|
|
# If it doesn't match but it's a whitelist ignore.
|
|
return
|
|
|
|
if not self.func:
|
|
return True
|
|
|
|
# Return the result of func directly as it may need to be awaited
|
|
return self.func(event)
|
|
|
|
|
|
class EventCommon(ChatGetter, abc.ABC):
|
|
"""
|
|
Intermediate class with common things to all events.
|
|
|
|
Remember that this class implements `ChatGetter
|
|
<telethon.tl.custom.chatgetter.ChatGetter>` which
|
|
means you have access to all chat properties and methods.
|
|
|
|
In addition, you can access the `original_update`
|
|
field which contains the original :tl:`Update`.
|
|
"""
|
|
_event_name = 'Event'
|
|
|
|
def __init__(self, chat_peer=None, msg_id=None, broadcast=None):
|
|
super().__init__(chat_peer, broadcast=broadcast)
|
|
self._entities = {}
|
|
self._client = None
|
|
self._message_id = msg_id
|
|
self.original_update = None
|
|
|
|
def _set_client(self, client):
|
|
"""
|
|
Setter so subclasses can act accordingly when the client is set.
|
|
"""
|
|
# TODO Nuke
|
|
self._client = client
|
|
if self._chat_peer:
|
|
self._chat, self._input_chat = utils._get_entity_pair(self.chat_id, self._entities)
|
|
else:
|
|
self._chat = self._input_chat = None
|
|
|
|
@property
|
|
def client(self):
|
|
"""
|
|
The `telethon.TelegramClient` that created this event.
|
|
"""
|
|
return self._client
|
|
|
|
def __str__(self):
|
|
return _tl.TLObject.pretty_format(self.to_dict())
|
|
|
|
def stringify(self):
|
|
return _tl.TLObject.pretty_format(self.to_dict(), indent=0)
|
|
|
|
def to_dict(self):
|
|
d = {k: v for k, v in self.__dict__.items() if k[0] != '_'}
|
|
d['_'] = self._event_name
|
|
return d
|
|
|
|
|
|
def name_inner_event(cls):
|
|
"""Decorator to rename cls.Event 'Event' as 'cls.Event'"""
|
|
if hasattr(cls, 'Event'):
|
|
cls.Event._event_name = '{}.Event'.format(cls.__name__)
|
|
else:
|
|
warnings.warn('Class {} does not have a inner Event'.format(cls))
|
|
return cls
|