python-telegram-bot/tests/test_callbackcontext.py

229 lines
8.5 KiB
Python
Raw Normal View History

#!/usr/bin/env python
#
# A library that provides a Python interface to the Telegram Bot API
2022-01-03 08:15:18 +01:00
# Copyright (C) 2015-2022
# Leandro Toledo de Souza <devs@python-telegram-bot.org>
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU Lesser Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU Lesser Public License for more details.
#
# You should have received a copy of the GNU Lesser Public License
# along with this program. If not, see [http://www.gnu.org/licenses/].
import pytest
2021-06-06 11:48:48 +02:00
from telegram import (
Update,
Message,
Chat,
User,
Bot,
InlineKeyboardMarkup,
InlineKeyboardButton,
CallbackQuery,
)
from telegram.ext import CallbackContext
from telegram.error import TelegramError
"""
CallbackContext.refresh_data is tested in TestBasePersistence
"""
class TestCallbackContext:
2021-08-29 18:15:04 +02:00
def test_slot_behaviour(self, dp, mro_slots, recwarn):
c = CallbackContext(dp)
for attr in c.__slots__:
assert getattr(c, attr, 'err') != 'err', f"got extra slot '{attr}'"
assert not c.__dict__, f"got missing slot(s): {c.__dict__}"
assert len(mro_slots(c)) == len(set(mro_slots(c))), "duplicate slot"
c.args = c.args
assert len(recwarn) == 0, recwarn.list
2021-08-29 18:15:04 +02:00
def test_from_job(self, dp):
job = dp.job_queue.run_once(lambda x: x, 10)
2021-08-29 18:15:04 +02:00
callback_context = CallbackContext.from_job(job, dp)
assert callback_context.job is job
assert callback_context.chat_data is None
assert callback_context.user_data is None
2021-08-29 18:15:04 +02:00
assert callback_context.bot_data is dp.bot_data
assert callback_context.bot is dp.bot
assert callback_context.job_queue is dp.job_queue
assert callback_context.update_queue is dp.update_queue
2021-08-29 18:15:04 +02:00
def test_from_update(self, dp):
update = Update(
0, message=Message(0, None, Chat(1, 'chat'), from_user=User(1, 'user', False))
)
2021-08-29 18:15:04 +02:00
callback_context = CallbackContext.from_update(update, dp)
assert callback_context.chat_data == {}
assert callback_context.user_data == {}
2021-08-29 18:15:04 +02:00
assert callback_context.bot_data is dp.bot_data
assert callback_context.bot is dp.bot
assert callback_context.job_queue is dp.job_queue
assert callback_context.update_queue is dp.update_queue
2021-08-29 18:15:04 +02:00
callback_context_same_user_chat = CallbackContext.from_update(update, dp)
callback_context.bot_data['test'] = 'bot'
callback_context.chat_data['test'] = 'chat'
callback_context.user_data['test'] = 'user'
assert callback_context_same_user_chat.bot_data is callback_context.bot_data
assert callback_context_same_user_chat.chat_data is callback_context.chat_data
assert callback_context_same_user_chat.user_data is callback_context.user_data
update_other_user_chat = Update(
0, message=Message(0, None, Chat(2, 'chat'), from_user=User(2, 'user', False))
)
2021-08-29 18:15:04 +02:00
callback_context_other_user_chat = CallbackContext.from_update(update_other_user_chat, dp)
assert callback_context_other_user_chat.bot_data is callback_context.bot_data
assert callback_context_other_user_chat.chat_data is not callback_context.chat_data
assert callback_context_other_user_chat.user_data is not callback_context.user_data
2021-08-29 18:15:04 +02:00
def test_from_update_not_update(self, dp):
callback_context = CallbackContext.from_update(None, dp)
assert callback_context.chat_data is None
assert callback_context.user_data is None
2021-08-29 18:15:04 +02:00
assert callback_context.bot_data is dp.bot_data
assert callback_context.bot is dp.bot
assert callback_context.job_queue is dp.job_queue
assert callback_context.update_queue is dp.update_queue
2021-08-29 18:15:04 +02:00
callback_context = CallbackContext.from_update('', dp)
assert callback_context.chat_data is None
assert callback_context.user_data is None
2021-08-29 18:15:04 +02:00
assert callback_context.bot_data is dp.bot_data
assert callback_context.bot is dp.bot
assert callback_context.job_queue is dp.job_queue
assert callback_context.update_queue is dp.update_queue
2021-08-29 18:15:04 +02:00
def test_from_error(self, dp):
error = TelegramError('test')
update = Update(
0, message=Message(0, None, Chat(1, 'chat'), from_user=User(1, 'user', False))
)
2021-08-29 18:15:04 +02:00
callback_context = CallbackContext.from_error(update, error, dp)
assert callback_context.error is error
assert callback_context.chat_data == {}
assert callback_context.user_data == {}
2021-08-29 18:15:04 +02:00
assert callback_context.bot_data is dp.bot_data
assert callback_context.bot is dp.bot
assert callback_context.job_queue is dp.job_queue
assert callback_context.update_queue is dp.update_queue
2020-10-04 17:20:33 +02:00
assert callback_context.async_args is None
assert callback_context.async_kwargs is None
2021-08-29 18:15:04 +02:00
def test_from_error_async_params(self, dp):
2020-10-04 17:20:33 +02:00
error = TelegramError('test')
args = [1, '2']
kwargs = {'one': 1, 2: 'two'}
callback_context = CallbackContext.from_error(
2021-08-29 18:15:04 +02:00
None, error, dp, async_args=args, async_kwargs=kwargs
)
2020-10-04 17:20:33 +02:00
assert callback_context.error is error
assert callback_context.async_args is args
assert callback_context.async_kwargs is kwargs
Update Filters, CommandHandler and MessageHandler (#1221) * update_filter attribute on filters Makes it possible to have filters work on an update instead of message, while keeping behavior for current filters * add update_type filter * Messagehandler rework - remove allow_edited (deprecated for a while) - set deprecated defaults to None - Raise deprecation warning when they're used - add sensible defaults for filters. - rework tests * Commandhandler rework * Remove deprecation test from new handler * Some tweaks per CR - rename update_types -> updates - added some clarification to docstrings * run webhook set test only on 3.6 on appveyor * update_filter attribute on filters Makes it possible to have filters work on an update instead of message, while keeping behavior for current filters * add update_type filter * Messagehandler rework - remove allow_edited (deprecated for a while) - set deprecated defaults to None - Raise deprecation warning when they're used - add sensible defaults for filters. - rework tests * Commandhandler rework * Remove deprecation test from new handler * Some tweaks per CR - rename update_types -> updates - added some clarification to docstrings * run webhook set test only on 3.6 on appveyor * Changes per CR * Update travis to build v12 * small doc update * try to make ci build version branches * doc for BaseFilter * Modify regexfilter and mergedfilter Now returns a list of match objects for every regexfilter * Change callbackcontext (+ docs) * integrate in CommandHandler and PrefixHandler * integrate in MessageHandler * cbqhandler, iqhandler and srhandler * make regexhandler a shell over MessageHandler And raise deprecationWarning on creation * clean up code and add some comments * Rework based on internal group feedback - use data_filter instead of regex_filter on BaseFilter - have these filters return a dict that is then updated onto CallbackContext instead of using a list is before - Add a .match property on CallbackContext that returns .matches[0] or None * Fix and add test for callbackcontext.match * Lots of documentation fixes and improvements [ci skip]
2019-02-13 12:07:25 +01:00
2021-08-29 18:15:04 +02:00
def test_match(self, dp):
callback_context = CallbackContext(dp)
Update Filters, CommandHandler and MessageHandler (#1221) * update_filter attribute on filters Makes it possible to have filters work on an update instead of message, while keeping behavior for current filters * add update_type filter * Messagehandler rework - remove allow_edited (deprecated for a while) - set deprecated defaults to None - Raise deprecation warning when they're used - add sensible defaults for filters. - rework tests * Commandhandler rework * Remove deprecation test from new handler * Some tweaks per CR - rename update_types -> updates - added some clarification to docstrings * run webhook set test only on 3.6 on appveyor * update_filter attribute on filters Makes it possible to have filters work on an update instead of message, while keeping behavior for current filters * add update_type filter * Messagehandler rework - remove allow_edited (deprecated for a while) - set deprecated defaults to None - Raise deprecation warning when they're used - add sensible defaults for filters. - rework tests * Commandhandler rework * Remove deprecation test from new handler * Some tweaks per CR - rename update_types -> updates - added some clarification to docstrings * run webhook set test only on 3.6 on appveyor * Changes per CR * Update travis to build v12 * small doc update * try to make ci build version branches * doc for BaseFilter * Modify regexfilter and mergedfilter Now returns a list of match objects for every regexfilter * Change callbackcontext (+ docs) * integrate in CommandHandler and PrefixHandler * integrate in MessageHandler * cbqhandler, iqhandler and srhandler * make regexhandler a shell over MessageHandler And raise deprecationWarning on creation * clean up code and add some comments * Rework based on internal group feedback - use data_filter instead of regex_filter on BaseFilter - have these filters return a dict that is then updated onto CallbackContext instead of using a list is before - Add a .match property on CallbackContext that returns .matches[0] or None * Fix and add test for callbackcontext.match * Lots of documentation fixes and improvements [ci skip]
2019-02-13 12:07:25 +01:00
assert callback_context.match is None
callback_context.matches = ['test', 'blah']
assert callback_context.match == 'test'
2021-08-29 18:15:04 +02:00
def test_data_assignment(self, dp):
update = Update(
0, message=Message(0, None, Chat(1, 'chat'), from_user=User(1, 'user', False))
)
2021-08-29 18:15:04 +02:00
callback_context = CallbackContext.from_update(update, dp)
with pytest.raises(AttributeError):
callback_context.bot_data = {"test": 123}
with pytest.raises(AttributeError):
callback_context.user_data = {}
with pytest.raises(AttributeError):
callback_context.chat_data = "test"
2021-08-29 18:15:04 +02:00
def test_dispatcher_attribute(self, dp):
callback_context = CallbackContext(dp)
assert callback_context.dispatcher == dp
2021-06-06 11:48:48 +02:00
2021-08-29 18:15:04 +02:00
def test_drop_callback_data_exception(self, bot, dp):
2021-06-06 11:48:48 +02:00
non_ext_bot = Bot(bot.token)
update = Update(
0, message=Message(0, None, Chat(1, 'chat'), from_user=User(1, 'user', False))
)
2021-08-29 18:15:04 +02:00
callback_context = CallbackContext.from_update(update, dp)
2021-06-06 11:48:48 +02:00
with pytest.raises(RuntimeError, match='This telegram.ext.ExtBot instance does not'):
callback_context.drop_callback_data(None)
try:
2021-08-29 18:15:04 +02:00
dp.bot = non_ext_bot
2021-06-06 11:48:48 +02:00
with pytest.raises(RuntimeError, match='telegram.Bot does not allow for'):
callback_context.drop_callback_data(None)
finally:
2021-08-29 18:15:04 +02:00
dp.bot = bot
2021-06-06 11:48:48 +02:00
2021-08-29 18:15:04 +02:00
def test_drop_callback_data(self, dp, monkeypatch, chat_id):
monkeypatch.setattr(dp.bot, 'arbitrary_callback_data', True)
2021-06-06 11:48:48 +02:00
update = Update(
0, message=Message(0, None, Chat(1, 'chat'), from_user=User(1, 'user', False))
)
2021-08-29 18:15:04 +02:00
callback_context = CallbackContext.from_update(update, dp)
dp.bot.send_message(
2021-06-06 11:48:48 +02:00
chat_id=chat_id,
text='test',
reply_markup=InlineKeyboardMarkup.from_button(
InlineKeyboardButton('test', callback_data='callback_data')
),
)
2021-08-29 18:15:04 +02:00
keyboard_uuid = dp.bot.callback_data_cache.persistence_data[0][0][0]
button_uuid = list(dp.bot.callback_data_cache.persistence_data[0][0][2])[0]
2021-06-06 11:48:48 +02:00
callback_data = keyboard_uuid + button_uuid
callback_query = CallbackQuery(
id='1',
from_user=None,
chat_instance=None,
data=callback_data,
)
2021-08-29 18:15:04 +02:00
dp.bot.callback_data_cache.process_callback_query(callback_query)
2021-06-06 11:48:48 +02:00
try:
2021-08-29 18:15:04 +02:00
assert len(dp.bot.callback_data_cache.persistence_data[0]) == 1
assert list(dp.bot.callback_data_cache.persistence_data[1]) == ['1']
2021-06-06 11:48:48 +02:00
callback_context.drop_callback_data(callback_query)
2021-08-29 18:15:04 +02:00
assert dp.bot.callback_data_cache.persistence_data == ([], {})
2021-06-06 11:48:48 +02:00
finally:
2021-08-29 18:15:04 +02:00
dp.bot.callback_data_cache.clear_callback_data()
dp.bot.callback_data_cache.clear_callback_queries()