2019-08-23 21:20:41 +02:00
|
|
|
#!/usr/bin/env python
|
|
|
|
# A library that provides a Python interface to the Telegram Bot API
|
2023-01-01 21:31:29 +01:00
|
|
|
# Copyright (C) 2015-2023
|
2019-08-23 21:20:41 +02:00
|
|
|
# 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/].
|
2022-05-05 09:27:54 +02:00
|
|
|
from datetime import datetime, timedelta, timezone
|
2020-09-27 12:59:48 +02:00
|
|
|
|
2022-05-05 09:27:54 +02:00
|
|
|
import pytest
|
2020-09-27 12:59:48 +02:00
|
|
|
|
2023-09-03 13:43:44 +02:00
|
|
|
from telegram import Chat, MessageEntity, Poll, PollAnswer, PollOption, User
|
2023-04-18 16:16:23 +02:00
|
|
|
from telegram._utils.datetime import UTC, to_timestamp
|
2022-04-30 22:18:11 +02:00
|
|
|
from telegram.constants import PollType
|
2023-09-03 13:43:44 +02:00
|
|
|
from telegram.warnings import PTBDeprecationWarning
|
2023-02-22 20:19:46 +01:00
|
|
|
from tests.auxil.slots import mro_slots
|
2019-08-23 21:20:41 +02:00
|
|
|
|
|
|
|
|
2023-02-11 10:45:17 +01:00
|
|
|
@pytest.fixture(scope="module")
|
2019-08-23 21:20:41 +02:00
|
|
|
def poll_option():
|
2023-02-11 10:45:17 +01:00
|
|
|
out = PollOption(text=TestPollOptionBase.text, voter_count=TestPollOptionBase.voter_count)
|
2022-12-15 15:00:36 +01:00
|
|
|
out._unfreeze()
|
|
|
|
return out
|
2019-08-23 21:20:41 +02:00
|
|
|
|
|
|
|
|
2023-02-11 10:45:17 +01:00
|
|
|
class TestPollOptionBase:
|
2019-08-23 21:20:41 +02:00
|
|
|
text = "test option"
|
|
|
|
voter_count = 3
|
|
|
|
|
2023-02-11 10:45:17 +01:00
|
|
|
|
|
|
|
class TestPollOptionWithoutRequest(TestPollOptionBase):
|
2023-02-22 20:19:46 +01:00
|
|
|
def test_slot_behaviour(self, poll_option):
|
2021-05-29 16:18:16 +02:00
|
|
|
for attr in poll_option.__slots__:
|
|
|
|
assert getattr(poll_option, attr, "err") != "err", f"got extra slot '{attr}'"
|
|
|
|
assert len(mro_slots(poll_option)) == len(set(mro_slots(poll_option))), "duplicate slot"
|
|
|
|
|
2019-08-23 21:20:41 +02:00
|
|
|
def test_de_json(self):
|
|
|
|
json_dict = {"text": self.text, "voter_count": self.voter_count}
|
|
|
|
poll_option = PollOption.de_json(json_dict, None)
|
2022-10-07 11:51:53 +02:00
|
|
|
assert poll_option.api_kwargs == {}
|
2019-08-23 21:20:41 +02:00
|
|
|
|
|
|
|
assert poll_option.text == self.text
|
|
|
|
assert poll_option.voter_count == self.voter_count
|
|
|
|
|
|
|
|
def test_to_dict(self, poll_option):
|
|
|
|
poll_option_dict = poll_option.to_dict()
|
|
|
|
|
|
|
|
assert isinstance(poll_option_dict, dict)
|
|
|
|
assert poll_option_dict["text"] == poll_option.text
|
|
|
|
assert poll_option_dict["voter_count"] == poll_option.voter_count
|
|
|
|
|
2020-07-14 21:33:56 +02:00
|
|
|
def test_equality(self):
|
|
|
|
a = PollOption("text", 1)
|
|
|
|
b = PollOption("text", 1)
|
|
|
|
c = PollOption("text_1", 1)
|
|
|
|
d = PollOption("text", 2)
|
|
|
|
e = Poll(123, "question", ["O1", "O2"], 1, False, True, Poll.REGULAR, True)
|
|
|
|
|
|
|
|
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)
|
|
|
|
|
2019-08-23 21:20:41 +02:00
|
|
|
|
2023-02-11 10:45:17 +01:00
|
|
|
@pytest.fixture(scope="module")
|
2020-03-29 09:52:30 +02:00
|
|
|
def poll_answer():
|
|
|
|
return PollAnswer(
|
2023-09-03 13:43:44 +02:00
|
|
|
TestPollAnswerBase.poll_id,
|
|
|
|
TestPollAnswerBase.option_ids,
|
|
|
|
TestPollAnswerBase.user,
|
|
|
|
TestPollAnswerBase.voter_chat,
|
2020-03-29 09:52:30 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
|
2023-02-11 10:45:17 +01:00
|
|
|
class TestPollAnswerBase:
|
2020-03-29 09:52:30 +02:00
|
|
|
poll_id = "id"
|
|
|
|
option_ids = [2]
|
2023-09-03 13:43:44 +02:00
|
|
|
user = User(1, "", False)
|
|
|
|
voter_chat = Chat(1, "")
|
2020-03-29 09:52:30 +02:00
|
|
|
|
2023-02-11 10:45:17 +01:00
|
|
|
|
|
|
|
class TestPollAnswerWithoutRequest(TestPollAnswerBase):
|
2020-03-29 09:52:30 +02:00
|
|
|
def test_de_json(self):
|
|
|
|
json_dict = {
|
|
|
|
"poll_id": self.poll_id,
|
|
|
|
"option_ids": self.option_ids,
|
2023-09-03 13:43:44 +02:00
|
|
|
"user": self.user.to_dict(),
|
|
|
|
"voter_chat": self.voter_chat.to_dict(),
|
2020-03-29 09:52:30 +02:00
|
|
|
}
|
|
|
|
poll_answer = PollAnswer.de_json(json_dict, None)
|
2022-10-07 11:51:53 +02:00
|
|
|
assert poll_answer.api_kwargs == {}
|
2020-03-29 09:52:30 +02:00
|
|
|
|
|
|
|
assert poll_answer.poll_id == self.poll_id
|
2022-12-15 15:00:36 +01:00
|
|
|
assert poll_answer.option_ids == tuple(self.option_ids)
|
2023-09-03 13:43:44 +02:00
|
|
|
assert poll_answer.user == self.user
|
|
|
|
assert poll_answer.voter_chat == self.voter_chat
|
2020-03-29 09:52:30 +02:00
|
|
|
|
|
|
|
def test_to_dict(self, poll_answer):
|
|
|
|
poll_answer_dict = poll_answer.to_dict()
|
|
|
|
|
|
|
|
assert isinstance(poll_answer_dict, dict)
|
|
|
|
assert poll_answer_dict["poll_id"] == poll_answer.poll_id
|
2022-12-15 15:00:36 +01:00
|
|
|
assert poll_answer_dict["option_ids"] == list(poll_answer.option_ids)
|
2023-09-03 13:43:44 +02:00
|
|
|
assert poll_answer_dict["user"] == poll_answer.user.to_dict()
|
|
|
|
assert poll_answer_dict["voter_chat"] == poll_answer.voter_chat.to_dict()
|
2020-03-29 09:52:30 +02:00
|
|
|
|
2020-07-14 21:33:56 +02:00
|
|
|
def test_equality(self):
|
2023-09-03 13:43:44 +02:00
|
|
|
a = PollAnswer(123, [2], self.user, self.voter_chat)
|
|
|
|
b = PollAnswer(123, [2], self.user, Chat(1, ""))
|
|
|
|
c = PollAnswer(123, [2], User(1, "first", False), self.voter_chat)
|
|
|
|
d = PollAnswer(123, [1, 2], self.user, self.voter_chat)
|
|
|
|
e = PollAnswer(456, [2], self.user, self.voter_chat)
|
|
|
|
f = PollOption("Text", 1)
|
2020-07-14 21:33:56 +02:00
|
|
|
|
|
|
|
assert a == b
|
|
|
|
assert hash(a) == hash(b)
|
|
|
|
|
2023-09-03 13:43:44 +02:00
|
|
|
assert a == c
|
|
|
|
assert hash(a) == hash(c)
|
2020-07-14 21:33:56 +02:00
|
|
|
|
|
|
|
assert a != d
|
|
|
|
assert hash(a) != hash(d)
|
|
|
|
|
|
|
|
assert a != e
|
|
|
|
assert hash(a) != hash(e)
|
|
|
|
|
2023-09-03 13:43:44 +02:00
|
|
|
assert a != f
|
|
|
|
assert hash(a) != hash(f)
|
|
|
|
|
|
|
|
def test_order_warning(self, recwarn):
|
|
|
|
expected_warning = (
|
|
|
|
"From v20.5 the order of `option_ids` and `user` is changed as the latter one"
|
|
|
|
" became optional. Please update your code to use the new order."
|
|
|
|
)
|
|
|
|
PollAnswer(123, [2], self.user, self.voter_chat)
|
|
|
|
assert len(recwarn) == 0
|
|
|
|
PollAnswer(123, self.user, [2], self.voter_chat)
|
|
|
|
assert len(recwarn) == 1
|
|
|
|
assert str(recwarn[0].message) == expected_warning
|
|
|
|
assert recwarn[0].category is PTBDeprecationWarning
|
|
|
|
assert recwarn[0].filename == __file__, "wrong stacklevel"
|
|
|
|
|
2020-03-29 09:52:30 +02:00
|
|
|
|
2023-02-11 10:45:17 +01:00
|
|
|
@pytest.fixture(scope="module")
|
2019-08-23 21:20:41 +02:00
|
|
|
def poll():
|
2022-12-15 15:00:36 +01:00
|
|
|
poll = Poll(
|
2023-02-11 10:45:17 +01:00
|
|
|
TestPollBase.id_,
|
|
|
|
TestPollBase.question,
|
|
|
|
TestPollBase.options,
|
|
|
|
TestPollBase.total_voter_count,
|
|
|
|
TestPollBase.is_closed,
|
|
|
|
TestPollBase.is_anonymous,
|
|
|
|
TestPollBase.type,
|
|
|
|
TestPollBase.allows_multiple_answers,
|
|
|
|
explanation=TestPollBase.explanation,
|
|
|
|
explanation_entities=TestPollBase.explanation_entities,
|
|
|
|
open_period=TestPollBase.open_period,
|
|
|
|
close_date=TestPollBase.close_date,
|
2020-03-29 09:52:30 +02:00
|
|
|
)
|
2022-12-15 15:00:36 +01:00
|
|
|
poll._unfreeze()
|
|
|
|
return poll
|
2019-08-23 21:20:41 +02:00
|
|
|
|
|
|
|
|
2023-02-11 10:45:17 +01:00
|
|
|
class TestPollBase:
|
2020-02-23 22:03:58 +01:00
|
|
|
id_ = "id"
|
2019-08-23 21:20:41 +02:00
|
|
|
question = "Test?"
|
|
|
|
options = [PollOption("test", 10), PollOption("test2", 11)]
|
2020-03-29 09:52:30 +02:00
|
|
|
total_voter_count = 0
|
2019-08-23 21:20:41 +02:00
|
|
|
is_closed = True
|
2020-03-29 09:52:30 +02:00
|
|
|
is_anonymous = False
|
|
|
|
type = Poll.REGULAR
|
|
|
|
allows_multiple_answers = True
|
2020-05-02 11:56:52 +02:00
|
|
|
explanation = (
|
|
|
|
b"\\U0001f469\\u200d\\U0001f469\\u200d\\U0001f467"
|
|
|
|
b"\\u200d\\U0001f467\\U0001f431http://google.com"
|
|
|
|
).decode("unicode-escape")
|
|
|
|
explanation_entities = [MessageEntity(13, 17, MessageEntity.URL)]
|
|
|
|
open_period = 42
|
2022-05-05 09:27:54 +02:00
|
|
|
close_date = datetime.now(timezone.utc)
|
2019-08-23 21:20:41 +02:00
|
|
|
|
2023-02-11 10:45:17 +01:00
|
|
|
|
|
|
|
class TestPollWithoutRequest(TestPollBase):
|
2020-09-27 12:59:48 +02:00
|
|
|
def test_de_json(self, bot):
|
2019-08-23 21:20:41 +02:00
|
|
|
json_dict = {
|
2020-02-23 22:03:58 +01:00
|
|
|
"id": self.id_,
|
2019-08-23 21:20:41 +02:00
|
|
|
"question": self.question,
|
|
|
|
"options": [o.to_dict() for o in self.options],
|
2020-03-29 09:52:30 +02:00
|
|
|
"total_voter_count": self.total_voter_count,
|
|
|
|
"is_closed": self.is_closed,
|
|
|
|
"is_anonymous": self.is_anonymous,
|
|
|
|
"type": self.type,
|
2020-05-02 11:56:52 +02:00
|
|
|
"allows_multiple_answers": self.allows_multiple_answers,
|
|
|
|
"explanation": self.explanation,
|
|
|
|
"explanation_entities": [self.explanation_entities[0].to_dict()],
|
|
|
|
"open_period": self.open_period,
|
|
|
|
"close_date": to_timestamp(self.close_date),
|
2019-08-23 21:20:41 +02:00
|
|
|
}
|
2020-09-27 12:59:48 +02:00
|
|
|
poll = Poll.de_json(json_dict, bot)
|
2022-10-07 11:51:53 +02:00
|
|
|
assert poll.api_kwargs == {}
|
2019-08-23 21:20:41 +02:00
|
|
|
|
2020-02-23 22:03:58 +01:00
|
|
|
assert poll.id == self.id_
|
2019-08-23 21:20:41 +02:00
|
|
|
assert poll.question == self.question
|
2022-12-15 15:00:36 +01:00
|
|
|
assert poll.options == tuple(self.options)
|
2019-08-23 21:20:41 +02:00
|
|
|
assert poll.options[0].text == self.options[0].text
|
|
|
|
assert poll.options[0].voter_count == self.options[0].voter_count
|
|
|
|
assert poll.options[1].text == self.options[1].text
|
|
|
|
assert poll.options[1].voter_count == self.options[1].voter_count
|
2020-03-29 09:52:30 +02:00
|
|
|
assert poll.total_voter_count == self.total_voter_count
|
2019-08-23 21:20:41 +02:00
|
|
|
assert poll.is_closed == self.is_closed
|
2020-03-29 09:52:30 +02:00
|
|
|
assert poll.is_anonymous == self.is_anonymous
|
|
|
|
assert poll.type == self.type
|
|
|
|
assert poll.allows_multiple_answers == self.allows_multiple_answers
|
2020-05-02 11:56:52 +02:00
|
|
|
assert poll.explanation == self.explanation
|
2022-12-15 15:00:36 +01:00
|
|
|
assert poll.explanation_entities == tuple(self.explanation_entities)
|
2020-05-02 11:56:52 +02:00
|
|
|
assert poll.open_period == self.open_period
|
2022-05-05 09:27:54 +02:00
|
|
|
assert abs(poll.close_date - self.close_date) < timedelta(seconds=1)
|
2020-05-02 11:56:52 +02:00
|
|
|
assert to_timestamp(poll.close_date) == to_timestamp(self.close_date)
|
2019-08-23 21:20:41 +02:00
|
|
|
|
2023-04-18 16:16:23 +02:00
|
|
|
def test_de_json_localization(self, tz_bot, bot, raw_bot):
|
|
|
|
json_dict = {
|
|
|
|
"id": self.id_,
|
|
|
|
"question": self.question,
|
|
|
|
"options": [o.to_dict() for o in self.options],
|
|
|
|
"total_voter_count": self.total_voter_count,
|
|
|
|
"is_closed": self.is_closed,
|
|
|
|
"is_anonymous": self.is_anonymous,
|
|
|
|
"type": self.type,
|
|
|
|
"allows_multiple_answers": self.allows_multiple_answers,
|
|
|
|
"explanation": self.explanation,
|
|
|
|
"explanation_entities": [self.explanation_entities[0].to_dict()],
|
|
|
|
"open_period": self.open_period,
|
|
|
|
"close_date": to_timestamp(self.close_date),
|
|
|
|
}
|
|
|
|
|
|
|
|
poll_raw = Poll.de_json(json_dict, raw_bot)
|
|
|
|
poll_bot = Poll.de_json(json_dict, bot)
|
|
|
|
poll_bot_tz = Poll.de_json(json_dict, tz_bot)
|
|
|
|
|
|
|
|
# comparing utcoffsets because comparing timezones is unpredicatable
|
|
|
|
poll_bot_tz_offset = poll_bot_tz.close_date.utcoffset()
|
|
|
|
tz_bot_offset = tz_bot.defaults.tzinfo.utcoffset(
|
|
|
|
poll_bot_tz.close_date.replace(tzinfo=None)
|
|
|
|
)
|
|
|
|
|
|
|
|
assert poll_raw.close_date.tzinfo == UTC
|
|
|
|
assert poll_bot.close_date.tzinfo == UTC
|
|
|
|
assert poll_bot_tz_offset == tz_bot_offset
|
|
|
|
|
2019-08-23 21:20:41 +02:00
|
|
|
def test_to_dict(self, poll):
|
|
|
|
poll_dict = poll.to_dict()
|
|
|
|
|
|
|
|
assert isinstance(poll_dict, dict)
|
|
|
|
assert poll_dict["id"] == poll.id
|
|
|
|
assert poll_dict["question"] == poll.question
|
|
|
|
assert poll_dict["options"] == [o.to_dict() for o in poll.options]
|
2020-03-29 09:52:30 +02:00
|
|
|
assert poll_dict["total_voter_count"] == poll.total_voter_count
|
2019-08-23 21:20:41 +02:00
|
|
|
assert poll_dict["is_closed"] == poll.is_closed
|
2020-03-29 09:52:30 +02:00
|
|
|
assert poll_dict["is_anonymous"] == poll.is_anonymous
|
|
|
|
assert poll_dict["type"] == poll.type
|
|
|
|
assert poll_dict["allows_multiple_answers"] == poll.allows_multiple_answers
|
2020-05-02 11:56:52 +02:00
|
|
|
assert poll_dict["explanation"] == poll.explanation
|
|
|
|
assert poll_dict["explanation_entities"] == [poll.explanation_entities[0].to_dict()]
|
|
|
|
assert poll_dict["open_period"] == poll.open_period
|
|
|
|
assert poll_dict["close_date"] == to_timestamp(poll.close_date)
|
|
|
|
|
2023-02-11 10:45:17 +01:00
|
|
|
def test_equality(self):
|
|
|
|
a = Poll(123, "question", ["O1", "O2"], 1, False, True, Poll.REGULAR, True)
|
|
|
|
b = Poll(123, "question", ["o1", "o2"], 1, True, False, Poll.REGULAR, True)
|
|
|
|
c = Poll(456, "question", ["o1", "o2"], 1, True, False, Poll.REGULAR, True)
|
|
|
|
d = PollOption("Text", 1)
|
|
|
|
|
|
|
|
assert a == b
|
|
|
|
assert hash(a) == hash(b)
|
|
|
|
|
|
|
|
assert a != c
|
|
|
|
assert hash(a) != hash(c)
|
|
|
|
|
|
|
|
assert a != d
|
|
|
|
assert hash(a) != hash(d)
|
|
|
|
|
2022-04-30 22:18:11 +02:00
|
|
|
def test_enum_init(self):
|
|
|
|
poll = Poll(
|
|
|
|
type="foo",
|
|
|
|
id="id",
|
|
|
|
question="question",
|
|
|
|
options=[],
|
|
|
|
total_voter_count=0,
|
|
|
|
is_closed=False,
|
|
|
|
is_anonymous=False,
|
|
|
|
allows_multiple_answers=False,
|
|
|
|
)
|
|
|
|
assert poll.type == "foo"
|
|
|
|
poll = Poll(
|
|
|
|
type=PollType.QUIZ,
|
|
|
|
id="id",
|
|
|
|
question="question",
|
|
|
|
options=[],
|
|
|
|
total_voter_count=0,
|
|
|
|
is_closed=False,
|
|
|
|
is_anonymous=False,
|
|
|
|
allows_multiple_answers=False,
|
|
|
|
)
|
|
|
|
assert poll.type is PollType.QUIZ
|
|
|
|
|
2020-05-02 11:56:52 +02:00
|
|
|
def test_parse_entity(self, poll):
|
|
|
|
entity = MessageEntity(type=MessageEntity.URL, offset=13, length=17)
|
|
|
|
poll.explanation_entities = [entity]
|
|
|
|
|
|
|
|
assert poll.parse_explanation_entity(entity) == "http://google.com"
|
|
|
|
|
2022-04-24 12:38:09 +02:00
|
|
|
with pytest.raises(RuntimeError, match="Poll has no"):
|
|
|
|
Poll(
|
|
|
|
"id",
|
|
|
|
"question",
|
|
|
|
[PollOption("text", voter_count=0)],
|
|
|
|
total_voter_count=0,
|
|
|
|
is_closed=False,
|
|
|
|
is_anonymous=False,
|
|
|
|
type=Poll.QUIZ,
|
|
|
|
allows_multiple_answers=False,
|
|
|
|
).parse_explanation_entity(entity)
|
|
|
|
|
2020-05-02 11:56:52 +02:00
|
|
|
def test_parse_entities(self, poll):
|
|
|
|
entity = MessageEntity(type=MessageEntity.URL, offset=13, length=17)
|
|
|
|
entity_2 = MessageEntity(type=MessageEntity.BOLD, offset=13, length=1)
|
|
|
|
poll.explanation_entities = [entity_2, entity]
|
|
|
|
|
|
|
|
assert poll.parse_explanation_entities(MessageEntity.URL) == {entity: "http://google.com"}
|
|
|
|
assert poll.parse_explanation_entities() == {entity: "http://google.com", entity_2: "h"}
|