Telethon/telethon/_events/inlinequery.py

247 lines
8.7 KiB
Python
Raw Normal View History

2018-07-15 12:31:14 +03:00
import inspect
import re
import asyncio
from .common import EventBuilder, EventCommon, name_inner_event
2021-09-12 14:27:13 +03:00
from .._misc import utils
from .. import _tl
2021-09-12 17:58:06 +03:00
from ..types import _custom
2018-07-15 12:31:14 +03:00
@name_inner_event
class InlineQuery(EventBuilder):
"""
Occurs whenever you sign in as a bot and a user
sends an inline query such as ``@bot query``.
2018-07-15 12:31:14 +03:00
Args:
users (`entity`, optional):
May be one or more entities (username/peer/etc.), preferably IDs.
By default, only inline queries from these users will be handled.
blacklist_users (`bool`, optional):
Whether to treat the users as a blacklist instead of
as a whitelist (default). This means that every chat
will be handled *except* those specified in ``users``
which will be ignored if ``blacklist_users=True``.
pattern (`str`, `callable`, `Pattern`, optional):
If set, only queries matching this pattern will be handled.
You can specify a regex-like string which will be matched
against the message, a callable function that returns `True`
2018-07-15 12:31:14 +03:00
if a message is acceptable, or a compiled regex pattern.
2020-02-20 12:18:26 +03:00
Example
.. code-block:: python
from telethon import events
@client.on(events.InlineQuery)
async def handler(event):
builder = event.builder
# Two options (convert user text to UPPERCASE or lowercase)
await event.answer([
builder.article('UPPERCASE', text=event.text.upper()),
builder.article('lowercase', text=event.text.lower()),
])
2018-07-15 12:31:14 +03:00
"""
def __init__(
self, users=None, *, blacklist_users=False, func=None, pattern=None):
super().__init__(users, blacklist_chats=blacklist_users, func=func)
2018-07-15 12:31:14 +03:00
if isinstance(pattern, str):
self.pattern = re.compile(pattern).match
elif not pattern or callable(pattern):
self.pattern = pattern
elif hasattr(pattern, 'match') and callable(pattern.match):
self.pattern = pattern.match
else:
raise TypeError('Invalid pattern type given')
2018-07-19 02:47:32 +03:00
@classmethod
2021-12-11 15:31:38 +03:00
def build(cls, update, others=None, self_id=None, *todo, **todo2):
if isinstance(update, _tl.UpdateBotInlineQuery):
2019-06-30 14:23:18 +03:00
return cls.Event(update)
2018-07-15 12:31:14 +03:00
def filter(self, event):
if self.pattern:
match = self.pattern(event.text)
if not match:
return
event.pattern_match = match
return super().filter(event)
2021-09-12 17:58:06 +03:00
class Event(EventCommon, _custom.sendergetter.SenderGetter):
2018-07-15 12:31:14 +03:00
"""
Represents the event of a new callback query.
Members:
2020-06-06 14:47:46 +03:00
query (:tl:`UpdateBotInlineQuery`):
The original :tl:`UpdateBotInlineQuery`.
2018-07-15 12:31:14 +03:00
2020-06-06 14:47:46 +03:00
Make sure to access the `text` property of the query if
you want the text rather than the actual query object.
2018-10-12 13:38:46 +03:00
2018-07-15 12:31:14 +03:00
pattern_match (`obj`, optional):
The resulting object from calling the passed ``pattern``
function, which is ``re.compile(...).match`` by default.
"""
def __init__(self, query):
super().__init__(chat_peer=_tl.PeerUser(query.user_id))
2021-09-12 17:58:06 +03:00
_custom.sendergetter.SenderGetter.__init__(self, query.user_id)
2018-07-15 12:31:14 +03:00
self.query = query
self.pattern_match = None
self._answered = False
2019-05-01 18:07:12 +03:00
def _set_client(self, client):
super()._set_client(client)
self._sender, self._input_sender = utils._get_entity_pair(self.sender_id, self._entities)
2018-07-15 12:31:14 +03:00
@property
def id(self):
"""
Returns the unique identifier for the query ID.
"""
return self.query.query_id
@property
def text(self):
"""
Returns the text the user used to make the inline query.
"""
return self.query.query
@property
def offset(self):
"""
2018-10-17 12:47:51 +03:00
The string the user's client used as an offset for the query.
This will either be empty or equal to offsets passed to `answer`.
2018-07-15 12:31:14 +03:00
"""
return self.query.offset
@property
def geo(self):
"""
If the user location is requested when using inline mode
and the user's device is able to send it, this will return
the :tl:`GeoPoint` with the position of the user.
"""
return self.query.geo
2018-07-15 12:31:14 +03:00
@property
def builder(self):
"""
2018-10-12 13:38:46 +03:00
Returns a new `InlineBuilder
<telethon.tl.custom.inlinebuilder.InlineBuilder>` instance.
2018-07-15 12:31:14 +03:00
"""
2022-01-24 23:09:51 +03:00
return _custom.InlineBuilder(self._client)
2018-07-15 12:31:14 +03:00
async def answer(
self, results=None, cache_time=0, *,
gallery=False, next_offset=None, private=False,
2018-07-15 12:31:14 +03:00
switch_pm=None, switch_pm_param=''):
"""
Answers the inline query with the given results.
See the documentation for `builder` to know what kind of answers
can be given.
2018-07-15 12:31:14 +03:00
Args:
results (`list`, optional):
A list of :tl:`InputBotInlineResult` to use.
You should use `builder` to create these:
2018-10-12 13:38:46 +03:00
.. code-block:: python
2018-07-15 12:31:14 +03:00
builder = inline.builder
r1 = builder.article('Be nice', text='Have a nice day')
r2 = builder.article('Be bad', text="I don't like you")
await inline.answer([r1, r2])
You can send up to 50 results as documented in
https://core.telegram.org/bots/api#answerinlinequery.
Sending more will raise ``ResultsTooMuchError``,
and you should consider using `next_offset` to
paginate them.
2018-07-15 12:31:14 +03:00
cache_time (`int`, optional):
For how long this result should be cached on
the user's client. Defaults to 0 for no cache.
gallery (`bool`, optional):
Whether the results should show as a gallery (grid) or not.
next_offset (`str`, optional):
The offset the client will send when the user scrolls the
results and it repeats the request.
2018-07-15 12:31:14 +03:00
private (`bool`, optional):
Whether the results should be cached by Telegram
(not private) or by the user's client (private).
switch_pm (`str`, optional):
If set, this text will be shown in the results
to allow the user to switch to private messages.
switch_pm_param (`str`, optional):
Optional parameter to start the bot with if
`switch_pm` was used.
2019-05-09 13:24:37 +03:00
Example:
.. code-block:: python
@bot.on(events.InlineQuery)
async def handler(event):
builder = event.builder
rev_text = event.text[::-1]
await event.answer([
builder.article('Reverse text', text=rev_text),
builder.photo('/path/to/photo.jpg')
])
2018-07-15 12:31:14 +03:00
"""
if self._answered:
return
if results:
futures = [self._as_future(x) for x in results]
await asyncio.wait(futures)
# All futures will be in the `done` *set* that `wait` returns.
#
# Precisely because it's a `set` and not a `list`, it
# will not preserve the order, but since all futures
# completed we can use our original, ordered `list`.
results = [x.result() for x in futures]
else:
results = []
2018-07-15 12:31:14 +03:00
if switch_pm:
switch_pm = _tl.InlineBotSwitchPM(switch_pm, switch_pm_param)
2018-07-15 12:31:14 +03:00
return await self._client(
_tl.fn.messages.SetInlineBotResults(
2018-07-15 12:31:14 +03:00
query_id=self.query.query_id,
results=results,
cache_time=cache_time,
gallery=gallery,
next_offset=next_offset,
2018-07-15 12:31:14 +03:00
private=private,
switch_pm=switch_pm
)
)
@staticmethod
def _as_future(obj):
2018-07-15 12:31:14 +03:00
if inspect.isawaitable(obj):
return asyncio.ensure_future(obj)
2018-07-15 12:31:14 +03:00
f = asyncio.get_running_loop().create_future()
2018-07-15 12:31:14 +03:00
f.set_result(obj)
return f