Added conversation timeout in ConversationHandler (#895)

This commit is contained in:
Evgen 2018-03-01 14:34:47 +05:00 committed by Noam Meltzer
parent b67ea7a691
commit 811369d1a0
2 changed files with 71 additions and 2 deletions

View file

@ -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)

View file

@ -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