2016-04-27 00:08:35 +02:00
|
|
|
#!/usr/bin/env python
|
2016-04-26 23:42:50 +02:00
|
|
|
#
|
|
|
|
# A library that provides a Python interface to the Telegram Bot API
|
2018-01-04 16:16:06 +01:00
|
|
|
# Copyright (C) 2015-2018
|
2016-04-26 23:42:50 +02:00
|
|
|
# Leandro Toledo de Souza <devs@python-telegram-bot.org>
|
|
|
|
#
|
|
|
|
# This program is free software: you can redistribute it and/or modify
|
2017-08-11 23:58:41 +02:00
|
|
|
# it under the terms of the GNU Lesser Public License as published by
|
2016-04-26 23:42:50 +02:00
|
|
|
# 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
|
2017-08-11 23:58:41 +02:00
|
|
|
# GNU Lesser Public License for more details.
|
2016-04-26 23:42:50 +02:00
|
|
|
#
|
2017-08-11 23:58:41 +02:00
|
|
|
# You should have received a copy of the GNU Lesser Public License
|
2016-04-26 23:42:50 +02:00
|
|
|
# along with this program. If not, see [http://www.gnu.org/licenses/].
|
2017-08-11 23:58:41 +02:00
|
|
|
import datetime
|
2016-04-26 23:42:50 +02:00
|
|
|
|
2017-08-11 23:58:41 +02:00
|
|
|
import pytest
|
2016-04-26 23:42:50 +02:00
|
|
|
|
2018-03-16 23:41:48 +01:00
|
|
|
from telegram import Message, User, Chat, MessageEntity, Document
|
2016-10-15 22:59:41 +02:00
|
|
|
from telegram.ext import Filters, BaseFilter
|
2018-03-15 05:59:27 +01:00
|
|
|
import re
|
2017-04-23 23:22:05 +02:00
|
|
|
|
2017-06-20 09:49:01 +02:00
|
|
|
|
2017-08-11 23:58:41 +02:00
|
|
|
@pytest.fixture(scope='function')
|
|
|
|
def message():
|
2017-09-01 08:40:05 +02:00
|
|
|
return Message(0, User(0, 'Testuser', False), datetime.datetime.now(), Chat(0, 'private'))
|
2017-08-11 23:58:41 +02:00
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture(scope='function',
|
|
|
|
params=MessageEntity.ALL_TYPES)
|
|
|
|
def message_entity(request):
|
|
|
|
return MessageEntity(request.param, 0, 0, url='', user='')
|
|
|
|
|
|
|
|
|
|
|
|
class TestFilters(object):
|
|
|
|
def test_filters_all(self, message):
|
|
|
|
assert Filters.all(message)
|
|
|
|
|
|
|
|
def test_filters_text(self, message):
|
|
|
|
message.text = 'test'
|
|
|
|
assert Filters.text(message)
|
|
|
|
message.text = '/test'
|
|
|
|
assert not Filters.text(message)
|
|
|
|
|
|
|
|
def test_filters_command(self, message):
|
|
|
|
message.text = 'test'
|
|
|
|
assert not Filters.command(message)
|
|
|
|
message.text = '/test'
|
|
|
|
assert Filters.command(message)
|
|
|
|
|
2018-03-15 05:59:27 +01:00
|
|
|
def test_filters_regex(self, message):
|
|
|
|
message.text = '/start deep-linked param'
|
|
|
|
assert Filters.regex(r'deep-linked param')(message)
|
|
|
|
message.text = '/help'
|
|
|
|
assert Filters.regex(r'help')(message)
|
|
|
|
message.text = '/help'
|
|
|
|
assert Filters.regex('help')(message)
|
|
|
|
|
|
|
|
message.text = 'test'
|
|
|
|
assert not Filters.regex(r'fail')(message)
|
|
|
|
assert Filters.regex(r'test')(message)
|
|
|
|
assert Filters.regex(re.compile(r'test'))(message)
|
|
|
|
|
|
|
|
message.text = 'i love python'
|
|
|
|
assert Filters.regex(r'.\b[lo]{2}ve python')(message)
|
|
|
|
|
2018-08-26 20:39:01 +02:00
|
|
|
message.text = None
|
|
|
|
assert not Filters.regex(r'fail')(message)
|
|
|
|
|
2017-08-11 23:58:41 +02:00
|
|
|
def test_filters_reply(self, message):
|
2017-09-01 08:40:05 +02:00
|
|
|
another_message = Message(1, User(1, 'TestOther', False), datetime.datetime.now(),
|
2017-08-11 23:58:41 +02:00
|
|
|
Chat(0, 'private'))
|
|
|
|
message.text = 'test'
|
|
|
|
assert not Filters.reply(message)
|
|
|
|
message.reply_to_message = another_message
|
|
|
|
assert Filters.reply(message)
|
|
|
|
|
|
|
|
def test_filters_audio(self, message):
|
|
|
|
assert not Filters.audio(message)
|
|
|
|
message.audio = 'test'
|
|
|
|
assert Filters.audio(message)
|
|
|
|
|
|
|
|
def test_filters_document(self, message):
|
|
|
|
assert not Filters.document(message)
|
|
|
|
message.document = 'test'
|
|
|
|
assert Filters.document(message)
|
|
|
|
|
2018-03-16 23:41:48 +01:00
|
|
|
def test_filters_document_type(self, message):
|
|
|
|
message.document = Document("file_id", mime_type="application/vnd.android.package-archive")
|
|
|
|
assert Filters.document.apk(message)
|
|
|
|
assert Filters.document.application(message)
|
|
|
|
assert not Filters.document.doc(message)
|
|
|
|
assert not Filters.document.audio(message)
|
|
|
|
|
|
|
|
message.document.mime_type = "application/msword"
|
|
|
|
assert Filters.document.doc(message)
|
|
|
|
assert Filters.document.application(message)
|
|
|
|
assert not Filters.document.docx(message)
|
|
|
|
assert not Filters.document.audio(message)
|
|
|
|
|
|
|
|
message.document.mime_type = "application/vnd.openxmlformats-" \
|
|
|
|
"officedocument.wordprocessingml.document"
|
|
|
|
assert Filters.document.docx(message)
|
|
|
|
assert Filters.document.application(message)
|
|
|
|
assert not Filters.document.exe(message)
|
|
|
|
assert not Filters.document.audio(message)
|
|
|
|
|
|
|
|
message.document.mime_type = "application/x-ms-dos-executable"
|
|
|
|
assert Filters.document.exe(message)
|
|
|
|
assert Filters.document.application(message)
|
|
|
|
assert not Filters.document.docx(message)
|
|
|
|
assert not Filters.document.audio(message)
|
|
|
|
|
|
|
|
message.document.mime_type = "video/mp4"
|
|
|
|
assert Filters.document.gif(message)
|
|
|
|
assert Filters.document.video(message)
|
|
|
|
assert not Filters.document.jpg(message)
|
|
|
|
assert not Filters.document.text(message)
|
|
|
|
|
|
|
|
message.document.mime_type = "image/jpeg"
|
|
|
|
assert Filters.document.jpg(message)
|
|
|
|
assert Filters.document.image(message)
|
|
|
|
assert not Filters.document.mp3(message)
|
|
|
|
assert not Filters.document.video(message)
|
|
|
|
|
|
|
|
message.document.mime_type = "audio/mpeg"
|
|
|
|
assert Filters.document.mp3(message)
|
|
|
|
assert Filters.document.audio(message)
|
|
|
|
assert not Filters.document.pdf(message)
|
|
|
|
assert not Filters.document.image(message)
|
|
|
|
|
|
|
|
message.document.mime_type = "application/pdf"
|
|
|
|
assert Filters.document.pdf(message)
|
|
|
|
assert Filters.document.application(message)
|
|
|
|
assert not Filters.document.py(message)
|
|
|
|
assert not Filters.document.audio(message)
|
|
|
|
|
|
|
|
message.document.mime_type = "text/x-python"
|
|
|
|
assert Filters.document.py(message)
|
|
|
|
assert Filters.document.text(message)
|
|
|
|
assert not Filters.document.svg(message)
|
|
|
|
assert not Filters.document.application(message)
|
|
|
|
|
|
|
|
message.document.mime_type = "image/svg+xml"
|
|
|
|
assert Filters.document.svg(message)
|
|
|
|
assert Filters.document.image(message)
|
|
|
|
assert not Filters.document.txt(message)
|
|
|
|
assert not Filters.document.video(message)
|
|
|
|
|
|
|
|
message.document.mime_type = "text/plain"
|
|
|
|
assert Filters.document.txt(message)
|
|
|
|
assert Filters.document.text(message)
|
|
|
|
assert not Filters.document.targz(message)
|
|
|
|
assert not Filters.document.application(message)
|
|
|
|
|
|
|
|
message.document.mime_type = "application/x-compressed-tar"
|
|
|
|
assert Filters.document.targz(message)
|
|
|
|
assert Filters.document.application(message)
|
|
|
|
assert not Filters.document.wav(message)
|
|
|
|
assert not Filters.document.audio(message)
|
|
|
|
|
|
|
|
message.document.mime_type = "audio/x-wav"
|
|
|
|
assert Filters.document.wav(message)
|
|
|
|
assert Filters.document.audio(message)
|
|
|
|
assert not Filters.document.xml(message)
|
|
|
|
assert not Filters.document.image(message)
|
|
|
|
|
|
|
|
message.document.mime_type = "application/xml"
|
|
|
|
assert Filters.document.xml(message)
|
|
|
|
assert Filters.document.application(message)
|
|
|
|
assert not Filters.document.zip(message)
|
|
|
|
assert not Filters.document.audio(message)
|
|
|
|
|
|
|
|
message.document.mime_type = "application/zip"
|
|
|
|
assert Filters.document.zip(message)
|
|
|
|
assert Filters.document.application(message)
|
|
|
|
assert not Filters.document.apk(message)
|
|
|
|
assert not Filters.document.audio(message)
|
|
|
|
|
|
|
|
message.document.mime_type = "image/x-rgb"
|
|
|
|
assert not Filters.document.category("application/")(message)
|
|
|
|
assert not Filters.document.mime_type("application/x-sh")(message)
|
|
|
|
message.document.mime_type = "application/x-sh"
|
|
|
|
assert Filters.document.category("application/")(message)
|
|
|
|
assert Filters.document.mime_type("application/x-sh")(message)
|
|
|
|
|
Bot API 4.0 (#1168)
Telegram Passport (#1174):
- Add full support for telegram passport.
- New types: PassportData, PassportFile, EncryptedPassportElement, EncryptedCredentials, PassportElementError, PassportElementErrorDataField, PassportElementErrorFrontSide, PassportElementErrorReverseSide, PassportElementErrorSelfie, PassportElementErrorFile and PassportElementErrorFiles.
- New bot method: set_passport_data_errors
- New filter: Filters.passport_data
- Field passport_data field on Message
- PassportData is automagically decrypted when you specify your private key when creating Updater or Bot.
- PassportFiles is also automagically decrypted as you download/retrieve them.
- See new passportbot.py example for details on how to use, or go to our telegram passport wiki page for more info
- NOTE: Passport decryption requires new dependency `cryptography`.
Inputfile rework (#1184):
- Change how Inputfile is handled internally
- This allows support for specifying the thumbnails of photos and videos using the thumb= argument in the different send_ methods.
- Also allows Bot.send_media_group to actually finally send more than one media.
- Add thumb to Audio, Video and Videonote
- Add Bot.edit_message_media together with InputMediaAnimation, InputMediaAudio, and inputMediaDocument.
Other Bot API 4.0 changes:
- Add forusquare_type to Venue, InlineQueryResultVenue, InputVenueMessageContent, and Bot.send_venue. (#1170)
- Add vCard support by adding vcard field to Contact, InlineQueryResultContact, InputContactMessageContent, and Bot.send_contact. (#1166)
- Support new message entities: CASHTAG and PHONE_NUMBER. (#1179)
- Cashtag seems to be things like $USD and $GBP, but it seems telegram doesn't currently send them to bots.
- Phone number also seems to have limited support for now
- Add Bot.send_animation, add width, height, and duration to Animation, and add Filters.animation. (#1172)
Co-authored-by: Jasmin Bom <jsmnbom@gmail.com>
Co-authored-by: code1mountain <32801117+code1mountain@users.noreply.github.com>
Co-authored-by: Eldinnie <pieter.schutz+github@gmail.com>
Co-authored-by: mathefreak1 <mathefreak@hi2.in>
2018-08-29 14:18:58 +02:00
|
|
|
def test_filters_animation(self, message):
|
|
|
|
assert not Filters.animation(message)
|
|
|
|
message.animation = 'test'
|
|
|
|
assert Filters.animation(message)
|
|
|
|
|
2017-08-11 23:58:41 +02:00
|
|
|
def test_filters_photo(self, message):
|
|
|
|
assert not Filters.photo(message)
|
|
|
|
message.photo = 'test'
|
|
|
|
assert Filters.photo(message)
|
|
|
|
|
|
|
|
def test_filters_sticker(self, message):
|
|
|
|
assert not Filters.sticker(message)
|
|
|
|
message.sticker = 'test'
|
|
|
|
assert Filters.sticker(message)
|
|
|
|
|
|
|
|
def test_filters_video(self, message):
|
|
|
|
assert not Filters.video(message)
|
|
|
|
message.video = 'test'
|
|
|
|
assert Filters.video(message)
|
|
|
|
|
|
|
|
def test_filters_voice(self, message):
|
|
|
|
assert not Filters.voice(message)
|
|
|
|
message.voice = 'test'
|
|
|
|
assert Filters.voice(message)
|
|
|
|
|
2018-04-14 21:53:54 +02:00
|
|
|
def test_filters_video_note(self, message):
|
|
|
|
assert not Filters.video_note(message)
|
|
|
|
message.video_note = 'test'
|
|
|
|
assert Filters.video_note(message)
|
|
|
|
|
2017-08-11 23:58:41 +02:00
|
|
|
def test_filters_contact(self, message):
|
|
|
|
assert not Filters.contact(message)
|
|
|
|
message.contact = 'test'
|
|
|
|
assert Filters.contact(message)
|
|
|
|
|
|
|
|
def test_filters_location(self, message):
|
|
|
|
assert not Filters.location(message)
|
|
|
|
message.location = 'test'
|
|
|
|
assert Filters.location(message)
|
|
|
|
|
|
|
|
def test_filters_venue(self, message):
|
|
|
|
assert not Filters.venue(message)
|
|
|
|
message.venue = 'test'
|
|
|
|
assert Filters.venue(message)
|
|
|
|
|
|
|
|
def test_filters_status_update(self, message):
|
|
|
|
assert not Filters.status_update(message)
|
|
|
|
|
|
|
|
message.new_chat_members = ['test']
|
|
|
|
assert Filters.status_update(message)
|
|
|
|
assert Filters.status_update.new_chat_members(message)
|
|
|
|
message.new_chat_members = None
|
|
|
|
|
|
|
|
message.left_chat_member = 'test'
|
|
|
|
assert Filters.status_update(message)
|
|
|
|
assert Filters.status_update.left_chat_member(message)
|
|
|
|
message.left_chat_member = None
|
|
|
|
|
|
|
|
message.new_chat_title = 'test'
|
|
|
|
assert Filters.status_update(message)
|
|
|
|
assert Filters.status_update.new_chat_title(message)
|
|
|
|
message.new_chat_title = ''
|
|
|
|
|
|
|
|
message.new_chat_photo = 'test'
|
|
|
|
assert Filters.status_update(message)
|
|
|
|
assert Filters.status_update.new_chat_photo(message)
|
|
|
|
message.new_chat_photo = None
|
|
|
|
|
|
|
|
message.delete_chat_photo = True
|
|
|
|
assert Filters.status_update(message)
|
|
|
|
assert Filters.status_update.delete_chat_photo(message)
|
|
|
|
message.delete_chat_photo = False
|
|
|
|
|
|
|
|
message.group_chat_created = True
|
|
|
|
assert Filters.status_update(message)
|
|
|
|
assert Filters.status_update.chat_created(message)
|
|
|
|
message.group_chat_created = False
|
|
|
|
|
|
|
|
message.supergroup_chat_created = True
|
|
|
|
assert Filters.status_update(message)
|
|
|
|
assert Filters.status_update.chat_created(message)
|
|
|
|
message.supergroup_chat_created = False
|
|
|
|
|
|
|
|
message.channel_chat_created = True
|
|
|
|
assert Filters.status_update(message)
|
|
|
|
assert Filters.status_update.chat_created(message)
|
|
|
|
message.channel_chat_created = False
|
|
|
|
|
|
|
|
message.migrate_to_chat_id = 100
|
|
|
|
assert Filters.status_update(message)
|
|
|
|
assert Filters.status_update.migrate(message)
|
|
|
|
message.migrate_to_chat_id = 0
|
|
|
|
|
|
|
|
message.migrate_from_chat_id = 100
|
|
|
|
assert Filters.status_update(message)
|
|
|
|
assert Filters.status_update.migrate(message)
|
|
|
|
message.migrate_from_chat_id = 0
|
|
|
|
|
|
|
|
message.pinned_message = 'test'
|
|
|
|
assert Filters.status_update(message)
|
|
|
|
assert Filters.status_update.pinned_message(message)
|
|
|
|
message.pinned_message = None
|
|
|
|
|
2018-02-18 16:11:04 +01:00
|
|
|
message.connected_website = 'http://example.com/'
|
|
|
|
assert Filters.status_update(message)
|
|
|
|
assert Filters.status_update.connected_website(message)
|
|
|
|
message.connected_website = None
|
|
|
|
|
2017-08-11 23:58:41 +02:00
|
|
|
def test_filters_forwarded(self, message):
|
|
|
|
assert not Filters.forwarded(message)
|
|
|
|
message.forward_date = 'test'
|
|
|
|
assert Filters.forwarded(message)
|
|
|
|
|
|
|
|
def test_filters_game(self, message):
|
|
|
|
assert not Filters.game(message)
|
|
|
|
message.game = 'test'
|
|
|
|
assert Filters.game(message)
|
|
|
|
|
|
|
|
def test_entities_filter(self, message, message_entity):
|
|
|
|
message.entities = [message_entity]
|
|
|
|
assert Filters.entity(message_entity.type)(message)
|
|
|
|
|
|
|
|
message.entities = []
|
|
|
|
assert not Filters.entity(MessageEntity.MENTION)(message)
|
|
|
|
|
|
|
|
second = message_entity.to_dict()
|
|
|
|
second['type'] = 'bold'
|
|
|
|
second = MessageEntity.de_json(second, None)
|
|
|
|
message.entities = [message_entity, second]
|
|
|
|
assert Filters.entity(message_entity.type)(message)
|
2018-04-20 13:24:40 +02:00
|
|
|
assert not Filters.caption_entity(message_entity.type)(message)
|
|
|
|
|
|
|
|
def test_caption_entities_filter(self, message, message_entity):
|
|
|
|
message.caption_entities = [message_entity]
|
|
|
|
assert Filters.caption_entity(message_entity.type)(message)
|
|
|
|
|
|
|
|
message.caption_entities = []
|
|
|
|
assert not Filters.caption_entity(MessageEntity.MENTION)(message)
|
|
|
|
|
|
|
|
second = message_entity.to_dict()
|
|
|
|
second['type'] = 'bold'
|
|
|
|
second = MessageEntity.de_json(second, None)
|
|
|
|
message.caption_entities = [message_entity, second]
|
|
|
|
assert Filters.caption_entity(message_entity.type)(message)
|
|
|
|
assert not Filters.entity(message_entity.type)(message)
|
2017-08-11 23:58:41 +02:00
|
|
|
|
|
|
|
def test_private_filter(self, message):
|
|
|
|
assert Filters.private(message)
|
|
|
|
message.chat.type = 'group'
|
|
|
|
assert not Filters.private(message)
|
|
|
|
|
|
|
|
def test_group_filter(self, message):
|
|
|
|
assert not Filters.group(message)
|
|
|
|
message.chat.type = 'group'
|
|
|
|
assert Filters.group(message)
|
|
|
|
message.chat.type = 'supergroup'
|
|
|
|
assert Filters.group(message)
|
2017-06-19 19:53:44 +02:00
|
|
|
|
|
|
|
def test_filters_user(self):
|
2017-08-11 23:58:41 +02:00
|
|
|
with pytest.raises(ValueError, match='user_id or username'):
|
2017-06-22 12:20:11 +02:00
|
|
|
Filters.user(user_id=1, username='user')
|
2017-08-11 23:58:41 +02:00
|
|
|
with pytest.raises(ValueError, match='user_id or username'):
|
2017-06-20 09:49:01 +02:00
|
|
|
Filters.user()
|
|
|
|
|
2017-08-11 23:58:41 +02:00
|
|
|
def test_filters_user_id(self, message):
|
|
|
|
assert not Filters.user(user_id=1)(message)
|
|
|
|
message.from_user.id = 1
|
|
|
|
assert Filters.user(user_id=1)(message)
|
|
|
|
message.from_user.id = 2
|
|
|
|
assert Filters.user(user_id=[1, 2])(message)
|
|
|
|
assert not Filters.user(user_id=[3, 4])(message)
|
|
|
|
|
|
|
|
def test_filters_username(self, message):
|
|
|
|
assert not Filters.user(username='user')(message)
|
|
|
|
assert not Filters.user(username='Testuser')(message)
|
|
|
|
message.from_user.username = 'user'
|
|
|
|
assert Filters.user(username='@user')(message)
|
|
|
|
assert Filters.user(username='user')(message)
|
|
|
|
assert Filters.user(username=['user1', 'user', 'user2'])(message)
|
|
|
|
assert not Filters.user(username=['@username', '@user_2'])(message)
|
2016-10-15 22:59:41 +02:00
|
|
|
|
2017-08-11 23:58:41 +02:00
|
|
|
def test_filters_chat(self):
|
|
|
|
with pytest.raises(ValueError, match='chat_id or username'):
|
|
|
|
Filters.chat(chat_id=-1, username='chat')
|
|
|
|
with pytest.raises(ValueError, match='chat_id or username'):
|
|
|
|
Filters.chat()
|
|
|
|
|
|
|
|
def test_filters_chat_id(self, message):
|
|
|
|
assert not Filters.chat(chat_id=-1)(message)
|
|
|
|
message.chat.id = -1
|
|
|
|
assert Filters.chat(chat_id=-1)(message)
|
|
|
|
message.chat.id = -2
|
|
|
|
assert Filters.chat(chat_id=[-1, -2])(message)
|
|
|
|
assert not Filters.chat(chat_id=[-3, -4])(message)
|
|
|
|
|
|
|
|
def test_filters_chat_username(self, message):
|
|
|
|
assert not Filters.chat(username='chat')(message)
|
|
|
|
message.chat.username = 'chat'
|
|
|
|
assert Filters.chat(username='@chat')(message)
|
|
|
|
assert Filters.chat(username='chat')(message)
|
|
|
|
assert Filters.chat(username=['chat1', 'chat', 'chat2'])(message)
|
|
|
|
assert not Filters.chat(username=['@chat1', 'chat_2'])(message)
|
|
|
|
|
|
|
|
def test_filters_invoice(self, message):
|
|
|
|
assert not Filters.invoice(message)
|
|
|
|
message.invoice = 'test'
|
|
|
|
assert Filters.invoice(message)
|
|
|
|
|
|
|
|
def test_filters_successful_payment(self, message):
|
|
|
|
assert not Filters.successful_payment(message)
|
|
|
|
message.successful_payment = 'test'
|
|
|
|
assert Filters.successful_payment(message)
|
|
|
|
|
Bot API 4.0 (#1168)
Telegram Passport (#1174):
- Add full support for telegram passport.
- New types: PassportData, PassportFile, EncryptedPassportElement, EncryptedCredentials, PassportElementError, PassportElementErrorDataField, PassportElementErrorFrontSide, PassportElementErrorReverseSide, PassportElementErrorSelfie, PassportElementErrorFile and PassportElementErrorFiles.
- New bot method: set_passport_data_errors
- New filter: Filters.passport_data
- Field passport_data field on Message
- PassportData is automagically decrypted when you specify your private key when creating Updater or Bot.
- PassportFiles is also automagically decrypted as you download/retrieve them.
- See new passportbot.py example for details on how to use, or go to our telegram passport wiki page for more info
- NOTE: Passport decryption requires new dependency `cryptography`.
Inputfile rework (#1184):
- Change how Inputfile is handled internally
- This allows support for specifying the thumbnails of photos and videos using the thumb= argument in the different send_ methods.
- Also allows Bot.send_media_group to actually finally send more than one media.
- Add thumb to Audio, Video and Videonote
- Add Bot.edit_message_media together with InputMediaAnimation, InputMediaAudio, and inputMediaDocument.
Other Bot API 4.0 changes:
- Add forusquare_type to Venue, InlineQueryResultVenue, InputVenueMessageContent, and Bot.send_venue. (#1170)
- Add vCard support by adding vcard field to Contact, InlineQueryResultContact, InputContactMessageContent, and Bot.send_contact. (#1166)
- Support new message entities: CASHTAG and PHONE_NUMBER. (#1179)
- Cashtag seems to be things like $USD and $GBP, but it seems telegram doesn't currently send them to bots.
- Phone number also seems to have limited support for now
- Add Bot.send_animation, add width, height, and duration to Animation, and add Filters.animation. (#1172)
Co-authored-by: Jasmin Bom <jsmnbom@gmail.com>
Co-authored-by: code1mountain <32801117+code1mountain@users.noreply.github.com>
Co-authored-by: Eldinnie <pieter.schutz+github@gmail.com>
Co-authored-by: mathefreak1 <mathefreak@hi2.in>
2018-08-29 14:18:58 +02:00
|
|
|
def test_filters_passport_data(self, message):
|
|
|
|
assert not Filters.passport_data(message)
|
|
|
|
message.passport_data = 'test'
|
|
|
|
assert Filters.passport_data(message)
|
|
|
|
|
2017-08-11 23:58:41 +02:00
|
|
|
def test_language_filter_single(self, message):
|
|
|
|
message.from_user.language_code = 'en_US'
|
|
|
|
assert (Filters.language('en_US'))(message)
|
|
|
|
assert (Filters.language('en'))(message)
|
|
|
|
assert not (Filters.language('en_GB'))(message)
|
|
|
|
assert not (Filters.language('da'))(message)
|
|
|
|
message.from_user.language_code = 'da'
|
|
|
|
assert not (Filters.language('en_US'))(message)
|
|
|
|
assert not (Filters.language('en'))(message)
|
|
|
|
assert not (Filters.language('en_GB'))(message)
|
|
|
|
assert (Filters.language('da'))(message)
|
|
|
|
|
|
|
|
def test_language_filter_multiple(self, message):
|
|
|
|
f = Filters.language(['en_US', 'da'])
|
|
|
|
message.from_user.language_code = 'en_US'
|
|
|
|
assert f(message)
|
|
|
|
message.from_user.language_code = 'en_GB'
|
|
|
|
assert not f(message)
|
|
|
|
message.from_user.language_code = 'da'
|
|
|
|
assert f(message)
|
|
|
|
|
|
|
|
def test_and_filters(self, message):
|
|
|
|
message.text = 'test'
|
|
|
|
message.forward_date = True
|
|
|
|
assert (Filters.text & Filters.forwarded)(message)
|
|
|
|
message.text = '/test'
|
|
|
|
assert not (Filters.text & Filters.forwarded)(message)
|
|
|
|
message.text = 'test'
|
|
|
|
message.forward_date = None
|
|
|
|
assert not (Filters.text & Filters.forwarded)(message)
|
|
|
|
|
|
|
|
message.text = 'test'
|
|
|
|
message.forward_date = True
|
|
|
|
assert (Filters.text & Filters.forwarded & Filters.private)(message)
|
|
|
|
|
|
|
|
def test_or_filters(self, message):
|
|
|
|
message.text = 'test'
|
|
|
|
assert (Filters.text | Filters.status_update)(message)
|
|
|
|
message.group_chat_created = True
|
|
|
|
assert (Filters.text | Filters.status_update)(message)
|
|
|
|
message.text = None
|
|
|
|
assert (Filters.text | Filters.status_update)(message)
|
|
|
|
message.group_chat_created = False
|
|
|
|
assert not (Filters.text | Filters.status_update)(message)
|
|
|
|
|
|
|
|
def test_and_or_filters(self, message):
|
|
|
|
message.text = 'test'
|
|
|
|
message.forward_date = True
|
|
|
|
assert (Filters.text & (Filters.forwarded | Filters.status_update))(message)
|
|
|
|
message.forward_date = False
|
|
|
|
assert not (Filters.text & (Filters.forwarded | Filters.status_update))(message)
|
|
|
|
message.pinned_message = True
|
|
|
|
assert (Filters.text & (Filters.forwarded | Filters.status_update)(message))
|
|
|
|
|
|
|
|
assert str((Filters.text & (Filters.forwarded | Filters.entity(
|
|
|
|
MessageEntity.MENTION)))) == '<Filters.text and <Filters.forwarded or ' \
|
|
|
|
'Filters.entity(mention)>>'
|
|
|
|
|
|
|
|
def test_inverted_filters(self, message):
|
|
|
|
message.text = '/test'
|
|
|
|
assert Filters.command(message)
|
|
|
|
assert not (~Filters.command)(message)
|
|
|
|
message.text = 'test'
|
|
|
|
assert not Filters.command(message)
|
|
|
|
assert (~Filters.command)(message)
|
|
|
|
|
|
|
|
def test_inverted_and_filters(self, message):
|
|
|
|
message.text = '/test'
|
|
|
|
message.forward_date = 1
|
|
|
|
assert (Filters.forwarded & Filters.command)(message)
|
|
|
|
assert not (~Filters.forwarded & Filters.command)(message)
|
|
|
|
assert not (Filters.forwarded & ~Filters.command)(message)
|
|
|
|
assert not (~(Filters.forwarded & Filters.command))(message)
|
|
|
|
message.forward_date = None
|
|
|
|
assert not (Filters.forwarded & Filters.command)(message)
|
|
|
|
assert (~Filters.forwarded & Filters.command)(message)
|
|
|
|
assert not (Filters.forwarded & ~Filters.command)(message)
|
|
|
|
assert (~(Filters.forwarded & Filters.command))(message)
|
|
|
|
message.text = 'test'
|
|
|
|
assert not (Filters.forwarded & Filters.command)(message)
|
|
|
|
assert not (~Filters.forwarded & Filters.command)(message)
|
|
|
|
assert not (Filters.forwarded & ~Filters.command)(message)
|
|
|
|
assert (~(Filters.forwarded & Filters.command))(message)
|
|
|
|
|
|
|
|
def test_faulty_custom_filter(self, message):
|
2016-10-15 22:59:41 +02:00
|
|
|
class _CustomFilter(BaseFilter):
|
|
|
|
pass
|
|
|
|
|
|
|
|
custom = _CustomFilter()
|
|
|
|
|
2017-08-11 23:58:41 +02:00
|
|
|
with pytest.raises(NotImplementedError):
|
|
|
|
(custom & Filters.text)(message)
|
|
|
|
|
|
|
|
def test_custom_unnamed_filter(self, message):
|
2017-06-19 21:49:42 +02:00
|
|
|
class Unnamed(BaseFilter):
|
2017-08-11 23:58:41 +02:00
|
|
|
def filter(self, mes):
|
2017-06-19 21:49:42 +02:00
|
|
|
return True
|
|
|
|
|
|
|
|
unnamed = Unnamed()
|
2017-08-11 23:58:41 +02:00
|
|
|
assert str(unnamed) == Unnamed.__name__
|