mirror of
https://github.com/python-telegram-bot/python-telegram-bot.git
synced 2025-01-18 15:20:42 +01:00
Added conversation timeout in ConversationHandler (#895)
This commit is contained in:
parent
b67ea7a691
commit
811369d1a0
2 changed files with 71 additions and 2 deletions
|
@ -55,7 +55,7 @@ class ConversationHandler(Handler):
|
||||||
To change the state of conversation, the callback function of a handler must return the new
|
To change the state of conversation, the callback function of a handler must return the new
|
||||||
state after responding to the user. If it does not return anything (returning ``None`` by
|
state after responding to the user. If it does not return anything (returning ``None`` by
|
||||||
default), the state will not change. To end the conversation, the callback function must
|
default), the state will not change. To end the conversation, the callback function must
|
||||||
return :attr`END` or ``-1``.
|
return :attr:`END` or ``-1``.
|
||||||
|
|
||||||
Attributes:
|
Attributes:
|
||||||
entry_points (List[:class:`telegram.ext.Handler`]): A list of ``Handler`` objects that can
|
entry_points (List[:class:`telegram.ext.Handler`]): A list of ``Handler`` objects that can
|
||||||
|
@ -76,6 +76,9 @@ class ConversationHandler(Handler):
|
||||||
per_user (:obj:`bool`): Optional. If the conversationkey should contain the User's ID.
|
per_user (:obj:`bool`): Optional. If the conversationkey should contain the User's ID.
|
||||||
per_message (:obj:`bool`): Optional. If the conversationkey should contain the Message's
|
per_message (:obj:`bool`): Optional. If the conversationkey should contain the Message's
|
||||||
ID.
|
ID.
|
||||||
|
conversation_timeout (:obj:`float`|:obj:`datetime.timedelta`): Optional. When this handler
|
||||||
|
is inactive more than this timeout (in seconds), it will be automatically ended. If
|
||||||
|
this value is 0 (default), there will be no timeout.
|
||||||
|
|
||||||
Args:
|
Args:
|
||||||
entry_points (List[:class:`telegram.ext.Handler`]): A list of ``Handler`` objects that can
|
entry_points (List[:class:`telegram.ext.Handler`]): A list of ``Handler`` objects that can
|
||||||
|
@ -107,6 +110,9 @@ class ConversationHandler(Handler):
|
||||||
Default is ``True``.
|
Default is ``True``.
|
||||||
per_message (:obj:`bool`, optional): If the conversationkey should contain the Message's
|
per_message (:obj:`bool`, optional): If the conversationkey should contain the Message's
|
||||||
ID. Default is ``False``.
|
ID. Default is ``False``.
|
||||||
|
conversation_timeout (:obj:`float`|:obj:`datetime.timedelta`, optional): When this handler
|
||||||
|
is inactive more than this timeout (in seconds), it will be automatically ended. If
|
||||||
|
this value is 0 or None (default), there will be no timeout.
|
||||||
|
|
||||||
Raises:
|
Raises:
|
||||||
ValueError
|
ValueError
|
||||||
|
@ -124,7 +130,8 @@ class ConversationHandler(Handler):
|
||||||
timed_out_behavior=None,
|
timed_out_behavior=None,
|
||||||
per_chat=True,
|
per_chat=True,
|
||||||
per_user=True,
|
per_user=True,
|
||||||
per_message=False):
|
per_message=False,
|
||||||
|
conversation_timeout=None):
|
||||||
|
|
||||||
self.entry_points = entry_points
|
self.entry_points = entry_points
|
||||||
self.states = states
|
self.states = states
|
||||||
|
@ -136,7 +143,9 @@ class ConversationHandler(Handler):
|
||||||
self.per_user = per_user
|
self.per_user = per_user
|
||||||
self.per_chat = per_chat
|
self.per_chat = per_chat
|
||||||
self.per_message = per_message
|
self.per_message = per_message
|
||||||
|
self.conversation_timeout = conversation_timeout
|
||||||
|
|
||||||
|
self.timeout_jobs = dict()
|
||||||
self.conversations = dict()
|
self.conversations = dict()
|
||||||
self.current_conversation = None
|
self.current_conversation = None
|
||||||
self.current_handler = None
|
self.current_handler = None
|
||||||
|
@ -294,6 +303,16 @@ class ConversationHandler(Handler):
|
||||||
|
|
||||||
"""
|
"""
|
||||||
new_state = self.current_handler.handle_update(update, dispatcher)
|
new_state = self.current_handler.handle_update(update, dispatcher)
|
||||||
|
timeout_job = self.timeout_jobs.get(self.current_conversation)
|
||||||
|
|
||||||
|
if timeout_job is not None or new_state == self.END:
|
||||||
|
timeout_job.schedule_removal()
|
||||||
|
del self.timeout_jobs[self.current_conversation]
|
||||||
|
if self.conversation_timeout and new_state != self.END:
|
||||||
|
self.timeout_jobs[self.current_conversation] = dispatcher.job_queue.run_once(
|
||||||
|
self._trigger_timeout, self.conversation_timeout,
|
||||||
|
context=self.current_conversation
|
||||||
|
)
|
||||||
|
|
||||||
self.update_state(new_state, self.current_conversation)
|
self.update_state(new_state, self.current_conversation)
|
||||||
|
|
||||||
|
@ -309,3 +328,6 @@ class ConversationHandler(Handler):
|
||||||
|
|
||||||
elif new_state is not None:
|
elif new_state is not None:
|
||||||
self.conversations[key] = new_state
|
self.conversations[key] = new_state
|
||||||
|
|
||||||
|
def _trigger_timeout(self, bot, job):
|
||||||
|
self.update_state(self.END, job.context)
|
||||||
|
|
|
@ -294,3 +294,50 @@ class TestConversationHandler(object):
|
||||||
assert not handler.check_update(Update(0, message=message))
|
assert not handler.check_update(Update(0, message=message))
|
||||||
assert not handler.check_update(Update(0, pre_checkout_query=pre_checkout_query))
|
assert not handler.check_update(Update(0, pre_checkout_query=pre_checkout_query))
|
||||||
assert not handler.check_update(Update(0, shipping_query=shipping_query))
|
assert not handler.check_update(Update(0, shipping_query=shipping_query))
|
||||||
|
|
||||||
|
def test_conversation_timeout(self, dp, bot, user1):
|
||||||
|
handler = ConversationHandler(entry_points=self.entry_points, states=self.states,
|
||||||
|
fallbacks=self.fallbacks, conversation_timeout=0.5)
|
||||||
|
dp.add_handler(handler)
|
||||||
|
|
||||||
|
# Start state machine, then reach timeout
|
||||||
|
message = Message(0, user1, None, self.group, text='/start', bot=bot)
|
||||||
|
dp.process_update(Update(update_id=0, message=message))
|
||||||
|
assert handler.conversations.get((self.group.id, user1.id)) == self.THIRSTY
|
||||||
|
sleep(0.5)
|
||||||
|
dp.job_queue.tick()
|
||||||
|
assert handler.conversations.get((self.group.id, user1.id)) is None
|
||||||
|
|
||||||
|
# Start state machine, do something, then reach timeout
|
||||||
|
dp.process_update(Update(update_id=0, message=message))
|
||||||
|
assert handler.conversations.get((self.group.id, user1.id)) == self.THIRSTY
|
||||||
|
message.text = '/brew'
|
||||||
|
dp.job_queue.tick()
|
||||||
|
dp.process_update(Update(update_id=0, message=message))
|
||||||
|
assert handler.conversations.get((self.group.id, user1.id)) == self.BREWING
|
||||||
|
sleep(0.5)
|
||||||
|
dp.job_queue.tick()
|
||||||
|
assert handler.conversations.get((self.group.id, user1.id)) is None
|
||||||
|
|
||||||
|
def test_conversation_timeout_two_users(self, dp, bot, user1, user2):
|
||||||
|
handler = ConversationHandler(entry_points=self.entry_points, states=self.states,
|
||||||
|
fallbacks=self.fallbacks, conversation_timeout=0.5)
|
||||||
|
dp.add_handler(handler)
|
||||||
|
|
||||||
|
# Start state machine, do something as second user, then reach timeout
|
||||||
|
message = Message(0, user1, None, self.group, text='/start', bot=bot)
|
||||||
|
dp.process_update(Update(update_id=0, message=message))
|
||||||
|
assert handler.conversations.get((self.group.id, user1.id)) == self.THIRSTY
|
||||||
|
message.text = '/brew'
|
||||||
|
message.from_user = user2
|
||||||
|
dp.job_queue.tick()
|
||||||
|
dp.process_update(Update(update_id=0, message=message))
|
||||||
|
assert handler.conversations.get((self.group.id, user2.id)) is None
|
||||||
|
message.text = '/start'
|
||||||
|
dp.job_queue.tick()
|
||||||
|
dp.process_update(Update(update_id=0, message=message))
|
||||||
|
assert handler.conversations.get((self.group.id, user2.id)) == self.THIRSTY
|
||||||
|
sleep(0.5)
|
||||||
|
dp.job_queue.tick()
|
||||||
|
assert handler.conversations.get((self.group.id, user1.id)) is None
|
||||||
|
assert handler.conversations.get((self.group.id, user2.id)) is None
|
||||||
|
|
Loading…
Reference in a new issue