2024-02-08 21:42:00 +05:30
|
|
|
#!/usr/bin/env python
|
|
|
|
#
|
|
|
|
# A library that provides a Python interface to the Telegram Bot API
|
2024-02-19 22:06:25 +03:00
|
|
|
# Copyright (C) 2015-2024
|
2024-02-08 21:42:00 +05:30
|
|
|
# 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/].
|
2024-12-15 05:26:37 -04:00
|
|
|
import datetime as dtm
|
2024-02-08 21:42:00 +05:30
|
|
|
import inspect
|
|
|
|
from copy import deepcopy
|
|
|
|
|
|
|
|
import pytest
|
|
|
|
|
|
|
|
from telegram import (
|
|
|
|
Chat,
|
|
|
|
Dice,
|
|
|
|
MessageOrigin,
|
|
|
|
MessageOriginChannel,
|
|
|
|
MessageOriginChat,
|
|
|
|
MessageOriginHiddenUser,
|
|
|
|
MessageOriginUser,
|
|
|
|
User,
|
|
|
|
)
|
|
|
|
from telegram._utils.datetime import UTC, to_timestamp
|
|
|
|
from tests.auxil.slots import mro_slots
|
|
|
|
|
|
|
|
ignored = ["self", "api_kwargs"]
|
|
|
|
|
|
|
|
|
|
|
|
class MODefaults:
|
2024-12-15 05:26:37 -04:00
|
|
|
date: dtm.datetime = to_timestamp(dtm.datetime.utcnow())
|
2024-02-08 21:42:00 +05:30
|
|
|
chat = Chat(1, Chat.CHANNEL)
|
|
|
|
message_id = 123
|
|
|
|
author_signautre = "PTB"
|
|
|
|
sender_chat = Chat(1, Chat.CHANNEL)
|
|
|
|
sender_user_name = "PTB"
|
|
|
|
sender_user = User(1, "user", False)
|
|
|
|
|
|
|
|
|
|
|
|
def message_origin_channel():
|
|
|
|
return MessageOriginChannel(
|
|
|
|
MODefaults.date, MODefaults.chat, MODefaults.message_id, MODefaults.author_signautre
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
def message_origin_chat():
|
|
|
|
return MessageOriginChat(
|
|
|
|
MODefaults.date,
|
|
|
|
MODefaults.sender_chat,
|
|
|
|
MODefaults.author_signautre,
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
def message_origin_hidden_user():
|
|
|
|
return MessageOriginHiddenUser(MODefaults.date, MODefaults.sender_user_name)
|
|
|
|
|
|
|
|
|
|
|
|
def message_origin_user():
|
|
|
|
return MessageOriginUser(MODefaults.date, MODefaults.sender_user)
|
|
|
|
|
|
|
|
|
|
|
|
def make_json_dict(instance: MessageOrigin, include_optional_args: bool = False) -> dict:
|
|
|
|
"""Used to make the json dict which we use for testing de_json. Similar to iter_args()"""
|
|
|
|
json_dict = {"type": instance.type}
|
|
|
|
sig = inspect.signature(instance.__class__.__init__)
|
|
|
|
|
|
|
|
for param in sig.parameters.values():
|
|
|
|
if param.name in ignored: # ignore irrelevant params
|
|
|
|
continue
|
|
|
|
|
|
|
|
val = getattr(instance, param.name)
|
|
|
|
# Compulsory args-
|
|
|
|
if param.default is inspect.Parameter.empty:
|
|
|
|
if hasattr(val, "to_dict"): # convert the user object or any future ones to dict.
|
|
|
|
val = val.to_dict()
|
|
|
|
json_dict[param.name] = val
|
|
|
|
|
|
|
|
# If we want to test all args (for de_json)-
|
|
|
|
elif param.default is not inspect.Parameter.empty and include_optional_args:
|
|
|
|
json_dict[param.name] = val
|
|
|
|
return json_dict
|
|
|
|
|
|
|
|
|
|
|
|
def iter_args(
|
|
|
|
instance: MessageOrigin, de_json_inst: MessageOrigin, include_optional: bool = False
|
|
|
|
):
|
|
|
|
"""
|
|
|
|
We accept both the regular instance and de_json created instance and iterate over them for
|
|
|
|
easy one line testing later one.
|
|
|
|
"""
|
|
|
|
yield instance.type, de_json_inst.type # yield this here cause it's not available in sig.
|
|
|
|
|
|
|
|
sig = inspect.signature(instance.__class__.__init__)
|
|
|
|
for param in sig.parameters.values():
|
|
|
|
if param.name in ignored:
|
|
|
|
continue
|
|
|
|
inst_at, json_at = getattr(instance, param.name), getattr(de_json_inst, param.name)
|
2024-12-15 05:26:37 -04:00
|
|
|
if isinstance(json_at, dtm.datetime): # Convert datetime to int
|
2024-02-08 21:42:00 +05:30
|
|
|
json_at = to_timestamp(json_at)
|
|
|
|
if (
|
|
|
|
param.default is not inspect.Parameter.empty and include_optional
|
|
|
|
) or param.default is inspect.Parameter.empty:
|
|
|
|
yield inst_at, json_at
|
|
|
|
|
|
|
|
|
2024-08-07 15:56:46 -04:00
|
|
|
@pytest.fixture
|
2024-02-08 21:42:00 +05:30
|
|
|
def message_origin_type(request):
|
|
|
|
return request.param()
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"message_origin_type",
|
|
|
|
[
|
|
|
|
message_origin_channel,
|
|
|
|
message_origin_chat,
|
|
|
|
message_origin_hidden_user,
|
|
|
|
message_origin_user,
|
|
|
|
],
|
|
|
|
indirect=True,
|
|
|
|
)
|
|
|
|
class TestMessageOriginTypesWithoutRequest:
|
|
|
|
def test_slot_behaviour(self, message_origin_type):
|
|
|
|
inst = message_origin_type
|
|
|
|
for attr in inst.__slots__:
|
|
|
|
assert getattr(inst, attr, "err") != "err", f"got extra slot '{attr}'"
|
|
|
|
assert len(mro_slots(inst)) == len(set(mro_slots(inst))), "duplicate slot"
|
|
|
|
|
2024-09-13 19:10:09 +02:00
|
|
|
def test_de_json_required_args(self, offline_bot, message_origin_type):
|
2024-02-08 21:42:00 +05:30
|
|
|
cls = message_origin_type.__class__
|
2024-09-13 19:10:09 +02:00
|
|
|
assert cls.de_json({}, offline_bot) is None
|
2024-02-08 21:42:00 +05:30
|
|
|
|
|
|
|
json_dict = make_json_dict(message_origin_type)
|
2024-09-13 19:10:09 +02:00
|
|
|
const_message_origin = MessageOrigin.de_json(json_dict, offline_bot)
|
2024-02-08 21:42:00 +05:30
|
|
|
assert const_message_origin.api_kwargs == {}
|
|
|
|
|
|
|
|
assert isinstance(const_message_origin, MessageOrigin)
|
|
|
|
assert isinstance(const_message_origin, cls)
|
|
|
|
for msg_origin_type_at, const_msg_origin_at in iter_args(
|
|
|
|
message_origin_type, const_message_origin
|
|
|
|
):
|
|
|
|
assert msg_origin_type_at == const_msg_origin_at
|
|
|
|
|
2024-09-13 19:10:09 +02:00
|
|
|
def test_de_json_all_args(self, offline_bot, message_origin_type):
|
2024-02-08 21:42:00 +05:30
|
|
|
json_dict = make_json_dict(message_origin_type, include_optional_args=True)
|
2024-09-13 19:10:09 +02:00
|
|
|
const_message_origin = MessageOrigin.de_json(json_dict, offline_bot)
|
2024-02-08 21:42:00 +05:30
|
|
|
|
|
|
|
assert const_message_origin.api_kwargs == {}
|
|
|
|
|
|
|
|
assert isinstance(const_message_origin, MessageOrigin)
|
|
|
|
assert isinstance(const_message_origin, message_origin_type.__class__)
|
|
|
|
for msg_origin_type_at, const_msg_origin_at in iter_args(
|
|
|
|
message_origin_type, const_message_origin, True
|
|
|
|
):
|
|
|
|
assert msg_origin_type_at == const_msg_origin_at
|
|
|
|
|
2024-09-13 19:10:09 +02:00
|
|
|
def test_de_json_messageorigin_localization(
|
|
|
|
self, message_origin_type, tz_bot, offline_bot, raw_bot
|
|
|
|
):
|
2024-02-08 21:42:00 +05:30
|
|
|
json_dict = make_json_dict(message_origin_type, include_optional_args=True)
|
|
|
|
msgorigin_raw = MessageOrigin.de_json(json_dict, raw_bot)
|
2024-09-13 19:10:09 +02:00
|
|
|
msgorigin_bot = MessageOrigin.de_json(json_dict, offline_bot)
|
2024-02-08 21:42:00 +05:30
|
|
|
msgorigin_tz = MessageOrigin.de_json(json_dict, tz_bot)
|
|
|
|
|
|
|
|
# comparing utcoffsets because comparing timezones is unpredicatable
|
|
|
|
msgorigin_offset = msgorigin_tz.date.utcoffset()
|
|
|
|
tz_bot_offset = tz_bot.defaults.tzinfo.utcoffset(msgorigin_tz.date.replace(tzinfo=None))
|
|
|
|
|
|
|
|
assert msgorigin_raw.date.tzinfo == UTC
|
|
|
|
assert msgorigin_bot.date.tzinfo == UTC
|
|
|
|
assert msgorigin_offset == tz_bot_offset
|
|
|
|
|
2024-09-13 19:10:09 +02:00
|
|
|
def test_de_json_invalid_type(self, message_origin_type, offline_bot):
|
2024-02-08 21:42:00 +05:30
|
|
|
json_dict = {"type": "invalid", "date": MODefaults.date}
|
2024-09-13 19:10:09 +02:00
|
|
|
message_origin_type = MessageOrigin.de_json(json_dict, offline_bot)
|
2024-02-08 21:42:00 +05:30
|
|
|
|
|
|
|
assert type(message_origin_type) is MessageOrigin
|
|
|
|
assert message_origin_type.type == "invalid"
|
|
|
|
|
2024-09-13 19:10:09 +02:00
|
|
|
def test_de_json_subclass(self, message_origin_type, offline_bot, chat_id):
|
|
|
|
"""This makes sure that e.g. MessageOriginChat(data, offline_bot) never returns a
|
2024-02-08 21:42:00 +05:30
|
|
|
MessageOriginUser instance."""
|
|
|
|
cls = message_origin_type.__class__
|
|
|
|
json_dict = make_json_dict(message_origin_type, True)
|
2024-09-13 19:10:09 +02:00
|
|
|
assert type(cls.de_json(json_dict, offline_bot)) is cls
|
2024-02-08 21:42:00 +05:30
|
|
|
|
|
|
|
def test_to_dict(self, message_origin_type):
|
|
|
|
message_origin_dict = message_origin_type.to_dict()
|
|
|
|
|
|
|
|
assert isinstance(message_origin_dict, dict)
|
|
|
|
assert message_origin_dict["type"] == message_origin_type.type
|
|
|
|
assert message_origin_dict["date"] == message_origin_type.date
|
|
|
|
|
|
|
|
for slot in message_origin_type.__slots__: # additional verification for the optional args
|
|
|
|
if slot in ("chat", "sender_chat", "sender_user"):
|
|
|
|
assert (getattr(message_origin_type, slot)).to_dict() == message_origin_dict[slot]
|
|
|
|
continue
|
|
|
|
assert getattr(message_origin_type, slot) == message_origin_dict[slot]
|
|
|
|
|
|
|
|
def test_equality(self, message_origin_type):
|
|
|
|
a = MessageOrigin(type="type", date=MODefaults.date)
|
|
|
|
b = MessageOrigin(type="type", date=MODefaults.date)
|
|
|
|
c = message_origin_type
|
|
|
|
d = deepcopy(message_origin_type)
|
|
|
|
e = Dice(4, "emoji")
|
|
|
|
|
|
|
|
assert a == b
|
|
|
|
assert hash(a) == hash(b)
|
|
|
|
|
|
|
|
assert a != c
|
|
|
|
assert hash(a) != hash(c)
|
|
|
|
|
|
|
|
assert a != d
|
|
|
|
assert hash(a) != hash(d)
|
|
|
|
|
|
|
|
assert a != e
|
|
|
|
assert hash(a) != hash(e)
|
|
|
|
|
|
|
|
assert c == d
|
|
|
|
assert hash(c) == hash(d)
|
|
|
|
|
|
|
|
assert c != e
|
|
|
|
assert hash(c) != hash(e)
|