diff --git a/telegram/_chat.py b/telegram/_chat.py index 39bce15e0..4ac457c45 100644 --- a/telegram/_chat.py +++ b/telegram/_chat.py @@ -19,6 +19,7 @@ # along with this program. If not, see [http://www.gnu.org/licenses/]. """This module contains an object that represents a Telegram Chat.""" from datetime import datetime +from html import escape from typing import TYPE_CHECKING, ClassVar, List, Optional, Tuple, Union from telegram import constants @@ -30,6 +31,9 @@ from telegram._telegramobject import TelegramObject from telegram._utils import enum from telegram._utils.defaultvalue import DEFAULT_NONE from telegram._utils.types import DVInput, FileInput, JSONDict, ODVInput, ReplyMarkup +from telegram.helpers import escape_markdown +from telegram.helpers import mention_html as helpers_mention_html +from telegram.helpers import mention_markdown as helpers_mention_markdown if TYPE_CHECKING: from telegram import ( @@ -355,6 +359,107 @@ class Chat(TelegramObject): return super()._de_json(data=data, bot=bot, api_kwargs=api_kwargs) + def mention_markdown(self, name: str = None) -> str: + """ + Note: + :tg-const:`telegram.constants.ParseMode.MARKDOWN` is a legacy mode, retained by + Telegram for backward compatibility. You should use :meth:`mention_markdown_v2` + instead. + + .. versionadded:: 20.0 + + Args: + name (:obj:`str`): The name used as a link for the chat. Defaults to :attr:`full_name`. + + Returns: + :obj:`str`: The inline mention for the chat as markdown (version 1). + + Raises: + :exc:`TypeError`: If the chat is a private chat and neither the :paramref:`name` + nor the :attr:`first_name` is set, then throw an :exc:`TypeError`. + If the chat is a public chat and neither the :paramref:`name` nor the :attr:`title` + is set, then throw an :exc:`TypeError`. If chat is a private group chat, then + throw an :exc:`TypeError`. + + """ + if self.type == self.PRIVATE: + if name: + return helpers_mention_markdown(self.id, name) + if self.full_name: + return helpers_mention_markdown(self.id, self.full_name) + raise TypeError("Can not create a mention to a private chat without first name") + if self.username: + if name: + return f"[{name}]({self.link})" + if self.title: + return f"[{self.title}]({self.link})" + raise TypeError("Can not create a mention to a public chat without title") + raise TypeError("Can not create a mention to a private group chat") + + def mention_markdown_v2(self, name: str = None) -> str: + """ + .. versionadded:: 20.0 + + Args: + name (:obj:`str`): The name used as a link for the chat. Defaults to :attr:`full_name`. + + Returns: + :obj:`str`: The inline mention for the chat as markdown (version 2). + + Raises: + :exc:`TypeError`: If the chat is a private chat and neither the :paramref:`name` + nor the :attr:`first_name` is set, then throw an :exc:`TypeError`. + If the chat is a public chat and neither the :paramref:`name` nor the :attr:`title` + is set, then throw an :exc:`TypeError`. If chat is a private group chat, then + throw an :exc:`TypeError`. + + """ + if self.type == self.PRIVATE: + if name: + return helpers_mention_markdown(self.id, name, version=2) + if self.full_name: + return helpers_mention_markdown(self.id, self.full_name, version=2) + raise TypeError("Can not create a mention to a private chat without first name") + if self.username: + if name: + return f"[{escape_markdown(name, version=2)}]({self.link})" + if self.title: + return f"[{escape_markdown(self.title, version=2)}]({self.link})" + raise TypeError("Can not create a mention to a public chat without title") + raise TypeError("Can not create a mention to a private group chat") + + def mention_html(self, name: str = None) -> str: + """ + .. versionadded:: 20.0 + + Args: + name (:obj:`str`): The name used as a link for the chat. Defaults to :attr:`full_name`. + + Returns: + :obj:`str`: The inline mention for the chat as HTML. + + Raises: + :exc:`TypeError`: If the chat is a private chat and neither the :paramref:`name` + nor the :attr:`first_name` is set, then throw an :exc:`TypeError`. + If the chat is a public chat and neither the :paramref:`name` nor the :attr:`title` + is set, then throw an :exc:`TypeError`. If chat is a private group chat, then + throw an :exc:`TypeError`. + + """ + if self.type == self.PRIVATE: + if name: + return helpers_mention_html(self.id, name) + if self.full_name: + return helpers_mention_html(self.id, self.full_name) + raise TypeError("Can not create a mention to a private chat without first name") + if self.username: + if name: + return f'{escape(name)}' + if self.title: + return f'{escape(self.title)}' + raise TypeError("Can not create a mention to a public chat without title") + raise TypeError("Can not create a mention to a private group chat") + async def leave( self, *, diff --git a/tests/test_chat.py b/tests/test_chat.py index 1da9f4ae1..ddc9d3cdb 100644 --- a/tests/test_chat.py +++ b/tests/test_chat.py @@ -21,6 +21,7 @@ import pytest from telegram import Bot, Chat, ChatLocation, ChatPermissions, Location, User from telegram.constants import ChatAction, ChatType +from telegram.helpers import escape_markdown from tests.conftest import check_defaults_handling, check_shortcut_call, check_shortcut_signature @@ -880,6 +881,99 @@ class TestChat: monkeypatch.setattr(chat.get_bot(), "decline_chat_join_request", make_assertion) assert await chat.decline_join_request(user_id=42) + def test_mention_html(self): + with pytest.raises(TypeError, match="Can not create a mention to a private group chat"): + chat = Chat(id=1, type="foo") + chat.mention_html() + + expected = '{}' + chat = Chat( + id=1, type=Chat.PRIVATE, first_name="first\u2022name", last_name="last\u2022name" + ) + assert chat.mention_html("the_name*\u2022") == expected.format(chat.id, "the_name*\u2022") + assert chat.mention_html() == expected.format(chat.id, chat.full_name) + with pytest.raises( + TypeError, match="Can not create a mention to a private chat without first name" + ): + chat = Chat(id=1, type=Chat.PRIVATE, last_name="last\u2022name") + chat.mention_html() + + expected = '{}' + chat = Chat(id=1, type="foo", username="user\u2022name", title="\u2022title") + assert chat.mention_html("the_name*\u2022") == expected.format( + chat.username, "the_name*\u2022" + ) + assert chat.mention_html() == expected.format(chat.username, chat.title) + with pytest.raises( + TypeError, match="Can not create a mention to a public chat without title" + ): + chat = Chat(id=1, type="foo", username="user\u2022name") + chat.mention_html() + + def test_mention_markdown(self): + with pytest.raises(TypeError, match="Can not create a mention to a private group chat"): + chat = Chat(id=1, type="foo") + chat.mention_markdown() + + expected = "[{}](tg://user?id={})" + chat = Chat( + id=1, type=Chat.PRIVATE, first_name="first\u2022name", last_name="last\u2022name" + ) + assert chat.mention_markdown("the_name*\u2022") == expected.format( + "the_name*\u2022", chat.id + ) + assert chat.mention_markdown() == expected.format(chat.full_name, chat.id) + with pytest.raises( + TypeError, match="Can not create a mention to a private chat without first name" + ): + chat = Chat(id=1, type=Chat.PRIVATE, last_name="last\u2022name") + chat.mention_markdown() + + expected = "[{}](https://t.me/{})" + chat = Chat(id=1, type="foo", username="user\u2022name", title="\u2022title") + assert chat.mention_markdown("the_name*\u2022") == expected.format( + "the_name*\u2022", chat.username + ) + assert chat.mention_markdown() == expected.format(chat.title, chat.username) + with pytest.raises( + TypeError, match="Can not create a mention to a public chat without title" + ): + chat = Chat(id=1, type="foo", username="user\u2022name") + chat.mention_markdown() + + def test_mention_markdown_v2(self): + with pytest.raises(TypeError, match="Can not create a mention to a private group chat"): + chat = Chat(id=1, type="foo") + chat.mention_markdown_v2() + + expected = "[{}](tg://user?id={})" + chat = Chat(id=1, type=Chat.PRIVATE, first_name="first{name", last_name="last_name") + assert chat.mention_markdown_v2("the{name>\u2022") == expected.format( + "the\\{name\\>\u2022", chat.id + ) + assert chat.mention_markdown_v2() == expected.format( + escape_markdown(chat.full_name, version=2), chat.id + ) + with pytest.raises( + TypeError, match="Can not create a mention to a private chat without first name" + ): + chat = Chat(id=1, type=Chat.PRIVATE, last_name="last_name") + chat.mention_markdown_v2() + + expected = "[{}](https://t.me/{})" + chat = Chat(id=1, type="foo", username="user{name", title="{title") + assert chat.mention_markdown_v2("the{name>\u2022") == expected.format( + "the\\{name\\>\u2022", chat.username + ) + assert chat.mention_markdown_v2() == expected.format( + escape_markdown(chat.title, version=2), chat.username + ) + with pytest.raises( + TypeError, match="Can not create a mention to a public chat without title" + ): + chat = Chat(id=1, type="foo", username="user\u2022name") + chat.mention_markdown_v2() + def test_equality(self): a = Chat(self.id_, self.title, self.type_) b = Chat(self.id_, self.title, self.type_)