Refactor Warnings in ConversationHandler (#2755, #2784)

This commit is contained in:
Poolitzer 2021-11-06 15:19:21 +01:00 committed by Hinrich Mahler
parent 44f1ce3784
commit b852a6866d
3 changed files with 229 additions and 127 deletions

View file

@ -29,6 +29,7 @@ Hey! You're PRing? Cool! Please have a look at the below checklist. It's here to
- [ ] Link new and existing constants in docstrings instead of hard coded number and strings - [ ] Link new and existing constants in docstrings instead of hard coded number and strings
- [ ] Add new message types to `Message.effective_attachment` - [ ] Add new message types to `Message.effective_attachment`
- [ ] Added new handlers for new update types - [ ] Added new handlers for new update types
- [ ] Add the handlers to the warning loop in the `ConversationHandler`
- [ ] Added new filters for new message (sub)types - [ ] Added new filters for new message (sub)types
- [ ] Added or updated documentation for the changed class(es) and/or method(s) - [ ] Added or updated documentation for the changed class(es) and/or method(s)
- [ ] Updated the Bot API version number in all places: `README.rst` and `README_RAW.rst` (including the badge), as well as `telegram.constants.BOT_API_VERSION` - [ ] Updated the Bot API version number in all places: `README.rst` and `README_RAW.rst` (including the badge), as well as `telegram.constants.BOT_API_VERSION`

View file

@ -45,6 +45,9 @@ from telegram.ext import (
DispatcherHandlerStop, DispatcherHandlerStop,
Handler, Handler,
InlineQueryHandler, InlineQueryHandler,
StringCommandHandler,
StringRegexHandler,
TypeHandler,
) )
from telegram._utils.warnings import warn from telegram._utils.warnings import warn
from telegram.ext._utils.promise import Promise from telegram.ext._utils.promise import Promise
@ -239,6 +242,14 @@ class ConversationHandler(Handler[Update, CCT]):
map_to_parent: Dict[object, object] = None, map_to_parent: Dict[object, object] = None,
run_async: bool = False, run_async: bool = False,
): ):
# these imports need to be here because of circular import error otherwise
from telegram.ext import ( # pylint: disable=import-outside-toplevel
ShippingQueryHandler,
PreCheckoutQueryHandler,
PollHandler,
PollAnswerHandler,
)
self.run_async = run_async self.run_async = run_async
self._entry_points = entry_points self._entry_points = entry_points
@ -283,49 +294,77 @@ class ConversationHandler(Handler[Update, CCT]):
for state_handlers in states.values(): for state_handlers in states.values():
all_handlers.extend(state_handlers) all_handlers.extend(state_handlers)
if self.per_message: # this loop is going to warn the user about handlers which can work unexpected
# in conversations
# this link will be added to all warnings tied to per_* setting
per_faq_link = (
" Read this FAQ entry to learn more about the per_* settings: https://git.io/JtcyU."
)
for handler in all_handlers: for handler in all_handlers:
if not isinstance(handler, CallbackQueryHandler): if isinstance(handler, (StringCommandHandler, StringRegexHandler)):
warn(
"The `ConversationHandler` only handles updates of type `telegram.Update`. "
f"{handler.__class__.__name__} handles updates of type `str`.",
stacklevel=2,
)
elif isinstance(handler, TypeHandler) and not issubclass(handler.type, Update):
warn(
"The `ConversationHandler` only handles updates of type `telegram.Update`."
f" The TypeHandler is set to handle {handler.type.__name__}.",
stacklevel=2,
)
elif isinstance(handler, PollHandler):
warn(
"PollHandler will never trigger in a conversation since it has no information "
"about the chat or the user who voted in it. Do you mean the "
"`PollAnswerHandler`?",
stacklevel=2,
)
elif self.per_chat and (
isinstance(
handler,
(
ShippingQueryHandler,
InlineQueryHandler,
ChosenInlineResultHandler,
PreCheckoutQueryHandler,
PollAnswerHandler,
),
)
):
warn(
f"Updates handled by {handler.__class__.__name__} only have information about "
f"the user, so this handler won't ever be triggered if `per_chat=True`."
f"{per_faq_link}",
stacklevel=2,
)
elif self.per_message and not isinstance(handler, CallbackQueryHandler):
warn( warn(
"If 'per_message=True', all entry points, state handlers, and fallbacks" "If 'per_message=True', all entry points, state handlers, and fallbacks"
" must be 'CallbackQueryHandler', since no other handlers " " must be 'CallbackQueryHandler', since no other handlers "
"have a message context.", f"have a message context.{per_faq_link}",
stacklevel=2, stacklevel=2,
) )
break elif not self.per_message and isinstance(handler, CallbackQueryHandler):
else:
for handler in all_handlers:
if isinstance(handler, CallbackQueryHandler):
warn( warn(
"If 'per_message=False', 'CallbackQueryHandler' will not be " "If 'per_message=False', 'CallbackQueryHandler' will not be "
"tracked for every message.", f"tracked for every message.{per_faq_link}",
stacklevel=2, stacklevel=2,
) )
break
if self.per_chat: if self.conversation_timeout and isinstance(handler, self.__class__):
for handler in all_handlers:
if isinstance(handler, (InlineQueryHandler, ChosenInlineResultHandler)):
warn(
"If 'per_chat=True', 'InlineQueryHandler' can not be used, "
"since inline queries have no chat context.",
stacklevel=2,
)
break
if self.conversation_timeout:
for handler in all_handlers:
if isinstance(handler, self.__class__):
warn( warn(
"Using `conversation_timeout` with nested conversations is currently not " "Using `conversation_timeout` with nested conversations is currently not "
"supported. You can still try to use it, but it will likely behave " "supported. You can still try to use it, but it will likely behave "
"differently from what you expect.", "differently from what you expect.",
stacklevel=2, stacklevel=2,
) )
break
if self.run_async: if self.run_async:
for handler in all_handlers:
handler.run_async = True handler.run_async = True
@property @property

View file

@ -18,6 +18,7 @@
# along with this program. If not, see [http://www.gnu.org/licenses/]. # along with this program. If not, see [http://www.gnu.org/licenses/].
import logging import logging
from time import sleep from time import sleep
from warnings import filterwarnings
import pytest import pytest
from flaky import flaky from flaky import flaky
@ -45,7 +46,15 @@ from telegram.ext import (
DispatcherHandlerStop, DispatcherHandlerStop,
TypeHandler, TypeHandler,
JobQueue, JobQueue,
StringCommandHandler,
StringRegexHandler,
PollHandler,
ShippingQueryHandler,
ChosenInlineResultHandler,
PreCheckoutQueryHandler,
PollAnswerHandler,
) )
from telegram.warnings import PTBUserWarning
@pytest.fixture(scope='class') @pytest.fixture(scope='class')
@ -1325,59 +1334,88 @@ class TestConversationHandler:
assert handler.conversations.get((self.group.id, user1.id)) is None assert handler.conversations.get((self.group.id, user1.id)) is None
assert self.is_timeout assert self.is_timeout
def test_conversation_timeout_warning_only_shown_once(self, recwarn): def test_handlers_generate_warning(self, recwarn):
ConversationHandler( """
entry_points=self.entry_points, this function tests all handler + per_* setting combinations.
states={ """
self.THIRSTY: [
ConversationHandler(
entry_points=self.entry_points,
states={
self.BREWING: [CommandHandler('pourCoffee', self.drink)],
},
fallbacks=self.fallbacks,
)
],
self.DRINKING: [
ConversationHandler(
entry_points=self.entry_points,
states={
self.CODING: [CommandHandler('startCoding', self.code)],
},
fallbacks=self.fallbacks,
)
],
},
fallbacks=self.fallbacks,
conversation_timeout=100,
)
assert len(recwarn) == 1
assert str(recwarn[0].message) == (
"Using `conversation_timeout` with nested conversations is currently not "
"supported. You can still try to use it, but it will likely behave "
"differently from what you expect."
)
assert recwarn[0].filename == __file__, "incorrect stacklevel!"
def test_per_message_warning_is_only_shown_once(self, recwarn): # the warning message action needs to be set to always,
# otherwise only the first occurrence will be issued
filterwarnings(action="always", category=PTBUserWarning)
# this class doesn't do anything, its just not the Update class
class NotUpdate:
pass
# this conversation handler has the string, string_regex, Pollhandler and TypeHandler
# which should all generate a warning no matter the per_* setting. TypeHandler should
# not when the class is Update
ConversationHandler( ConversationHandler(
entry_points=self.entry_points, entry_points=[StringCommandHandler("code", self.code)],
states={ states={
self.THIRSTY: [CommandHandler('pourCoffee', self.drink)], self.BREWING: [
self.BREWING: [CommandHandler('startCoding', self.code)], StringRegexHandler("code", self.code),
PollHandler(self.code),
TypeHandler(NotUpdate, self.code),
],
}, },
fallbacks=self.fallbacks, fallbacks=[TypeHandler(Update, self.code)],
)
# these handlers should all raise a warning when per_chat is True
ConversationHandler(
entry_points=[ShippingQueryHandler(self.code)],
states={
self.BREWING: [
InlineQueryHandler(self.code),
PreCheckoutQueryHandler(self.code),
PollAnswerHandler(self.code),
],
},
fallbacks=[ChosenInlineResultHandler(self.code)],
per_chat=True,
)
# the CallbackQueryHandler should *not* raise when per_message is True,
# but any other one should
ConversationHandler(
entry_points=[CallbackQueryHandler(self.code)],
states={
self.BREWING: [CommandHandler("code", self.code)],
},
fallbacks=[CallbackQueryHandler(self.code)],
per_message=True, per_message=True,
) )
assert len(recwarn) == 1
assert str(recwarn[0].message) == (
"If 'per_message=True', all entry points, state handlers, and fallbacks"
" must be 'CallbackQueryHandler', since no other handlers"
" have a message context."
)
assert recwarn[0].filename == __file__, "incorrect stacklevel!"
def test_per_message_but_not_per_chat_warning(self, recwarn): # the CallbackQueryHandler should raise when per_message is False
ConversationHandler(
entry_points=[CommandHandler("code", self.code)],
states={
self.BREWING: [CommandHandler("code", self.code)],
},
fallbacks=[CallbackQueryHandler(self.code)],
per_message=False,
)
# adding a nested conv to a conversation with timeout should warn
child = ConversationHandler(
entry_points=[CommandHandler("code", self.code)],
states={
self.BREWING: [CommandHandler("code", self.code)],
},
fallbacks=[CommandHandler("code", self.code)],
)
ConversationHandler(
entry_points=[CommandHandler("code", self.code)],
states={
self.BREWING: [child],
},
fallbacks=[CommandHandler("code", self.code)],
conversation_timeout=42,
)
# If per_message is True, per_chat should also be True, since msg ids are not unique
ConversationHandler( ConversationHandler(
entry_points=[CallbackQueryHandler(self.code, "code")], entry_points=[CallbackQueryHandler(self.code, "code")],
states={ states={
@ -1387,52 +1425,76 @@ class TestConversationHandler:
per_message=True, per_message=True,
per_chat=False, per_chat=False,
) )
assert len(recwarn) == 1
# the overall number of handlers throwing a warning is 13
assert len(recwarn) == 13
# now we test the messages, they are raised in the order they are inserted
# into the conversation handler
assert str(recwarn[0].message) == ( assert str(recwarn[0].message) == (
"The `ConversationHandler` only handles updates of type `telegram.Update`. "
"StringCommandHandler handles updates of type `str`."
)
assert str(recwarn[1].message) == (
"The `ConversationHandler` only handles updates of type `telegram.Update`. "
"StringRegexHandler handles updates of type `str`."
)
assert str(recwarn[2].message) == (
"PollHandler will never trigger in a conversation since it has no information "
"about the chat or the user who voted in it. Do you mean the "
"`PollAnswerHandler`?"
)
assert str(recwarn[3].message) == (
"The `ConversationHandler` only handles updates of type `telegram.Update`. "
"The TypeHandler is set to handle NotUpdate."
)
per_faq_link = (
" Read this FAQ entry to learn more about the per_* settings: https://git.io/JtcyU."
)
assert str(recwarn[4].message) == (
"Updates handled by ShippingQueryHandler only have information about the user,"
" so this handler won't ever be triggered if `per_chat=True`." + per_faq_link
)
assert str(recwarn[5].message) == (
"Updates handled by ChosenInlineResultHandler only have information about the user,"
" so this handler won't ever be triggered if `per_chat=True`." + per_faq_link
)
assert str(recwarn[6].message) == (
"Updates handled by InlineQueryHandler only have information about the user,"
" so this handler won't ever be triggered if `per_chat=True`." + per_faq_link
)
assert str(recwarn[7].message) == (
"Updates handled by PreCheckoutQueryHandler only have information about the user,"
" so this handler won't ever be triggered if `per_chat=True`." + per_faq_link
)
assert str(recwarn[8].message) == (
"Updates handled by PollAnswerHandler only have information about the user,"
" so this handler won't ever be triggered if `per_chat=True`." + per_faq_link
)
assert str(recwarn[9].message) == (
"If 'per_message=True', all entry points, state handlers, and fallbacks must be "
"'CallbackQueryHandler', since no other handlers have a message context."
+ per_faq_link
)
assert str(recwarn[10].message) == (
"If 'per_message=False', 'CallbackQueryHandler' will not be tracked for "
"every message." + per_faq_link
)
assert str(recwarn[11].message) == (
"Using `conversation_timeout` with nested conversations is currently not "
"supported. You can still try to use it, but it will likely behave differently"
" from what you expect."
)
assert str(recwarn[12].message) == (
"If 'per_message=True' is used, 'per_chat=True' should also be used, " "If 'per_message=True' is used, 'per_chat=True' should also be used, "
"since message IDs are not globally unique." "since message IDs are not globally unique."
) )
assert recwarn[0].filename == __file__, "incorrect stacklevel!"
def test_per_message_false_warning_is_only_shown_once(self, recwarn): # this for loop checks if the correct stacklevel is used when generating the warning
ConversationHandler( for warning in recwarn:
entry_points=self.entry_points, assert warning.filename == __file__, "incorrect stacklevel!"
states={
self.THIRSTY: [CallbackQueryHandler(self.drink)],
self.BREWING: [CallbackQueryHandler(self.code)],
},
fallbacks=self.fallbacks,
per_message=False,
)
assert len(recwarn) == 1
assert str(recwarn[0].message) == (
"If 'per_message=False', 'CallbackQueryHandler' will not be "
"tracked for every message."
)
assert recwarn[0].filename == __file__, "incorrect stacklevel!"
def test_warnings_per_chat_is_only_shown_once(self, recwarn):
def hello(update, context):
return self.BREWING
def bye(update, context):
return ConversationHandler.END
ConversationHandler(
entry_points=self.entry_points,
states={
self.THIRSTY: [InlineQueryHandler(hello)],
self.BREWING: [InlineQueryHandler(bye)],
},
fallbacks=self.fallbacks,
per_chat=True,
)
assert len(recwarn) == 1
assert str(recwarn[0].message) == (
"If 'per_chat=True', 'InlineQueryHandler' can not be used,"
" since inline queries have no chat context."
)
assert recwarn[0].filename == __file__, "incorrect stacklevel!"
def test_nested_conversation_handler(self, dp, bot, user1, user2): def test_nested_conversation_handler(self, dp, bot, user1, user2):
self.nested_states[self.DRINKING] = [ self.nested_states[self.DRINKING] = [