2017-06-09 17:13:39 +03:00
|
|
|
"""
|
|
|
|
Utilities for working with the Telegram API itself (such as handy methods
|
|
|
|
to convert between an entity like an User, Chat, etc. into its Input version)
|
|
|
|
"""
|
2017-10-01 14:24:04 +03:00
|
|
|
import math
|
2016-11-13 16:51:50 +03:00
|
|
|
from mimetypes import add_type, guess_extension
|
2016-10-09 13:57:38 +03:00
|
|
|
|
2017-08-30 12:12:25 +03:00
|
|
|
from .tl import TLObject
|
2017-06-09 17:13:39 +03:00
|
|
|
from .tl.types import (
|
2017-06-15 18:03:59 +03:00
|
|
|
Channel, ChannelForbidden, Chat, ChatEmpty, ChatForbidden, ChatFull,
|
2017-07-10 17:09:20 +03:00
|
|
|
ChatPhoto, InputPeerChannel, InputPeerChat, InputPeerUser, InputPeerEmpty,
|
2017-07-07 10:48:06 +03:00
|
|
|
MessageMediaDocument, MessageMediaPhoto, PeerChannel, InputChannel,
|
2017-07-10 17:09:20 +03:00
|
|
|
UserEmpty, InputUser, InputUserEmpty, InputUserSelf, InputPeerSelf,
|
2017-09-25 14:43:03 +03:00
|
|
|
PeerChat, PeerUser, User, UserFull, UserProfilePhoto, Document,
|
|
|
|
MessageMediaContact, MessageMediaEmpty, MessageMediaGame, MessageMediaGeo,
|
|
|
|
MessageMediaUnsupported, MessageMediaVenue, InputMediaContact,
|
|
|
|
InputMediaDocument, InputMediaEmpty, InputMediaGame,
|
|
|
|
InputMediaGeoPoint, InputMediaPhoto, InputMediaVenue, InputDocument,
|
|
|
|
DocumentEmpty, InputDocumentEmpty, Message, GeoPoint, InputGeoPoint,
|
|
|
|
GeoPointEmpty, InputGeoPointEmpty, Photo, InputPhoto, PhotoEmpty,
|
|
|
|
InputPhotoEmpty, FileLocation, ChatPhotoEmpty, UserProfilePhotoEmpty,
|
|
|
|
FileLocationUnavailable, InputMediaUploadedDocument,
|
2017-10-08 14:45:14 +03:00
|
|
|
InputMediaUploadedPhoto, DocumentAttributeFilename, photos
|
|
|
|
)
|
2016-10-09 13:57:38 +03:00
|
|
|
|
|
|
|
|
|
|
|
def get_display_name(entity):
|
|
|
|
"""Gets the input peer for the given "entity" (user, chat or channel)
|
|
|
|
Returns None if it was not found"""
|
|
|
|
if isinstance(entity, User):
|
2017-06-16 10:11:49 +03:00
|
|
|
if entity.last_name and entity.first_name:
|
2016-10-09 13:57:38 +03:00
|
|
|
return '{} {}'.format(entity.first_name, entity.last_name)
|
2017-06-16 10:11:49 +03:00
|
|
|
elif entity.first_name:
|
|
|
|
return entity.first_name
|
|
|
|
elif entity.last_name:
|
|
|
|
return entity.last_name
|
|
|
|
else:
|
|
|
|
return '(No name)'
|
2016-10-09 13:57:38 +03:00
|
|
|
|
2017-10-13 12:38:12 +03:00
|
|
|
if isinstance(entity, (Chat, Channel)):
|
2016-10-09 13:57:38 +03:00
|
|
|
return entity.title
|
|
|
|
|
2017-05-23 10:45:48 +03:00
|
|
|
return '(unknown)'
|
|
|
|
|
2016-11-13 16:51:50 +03:00
|
|
|
# For some reason, .webp (stickers' format) is not registered
|
|
|
|
add_type('image/webp', '.webp')
|
|
|
|
|
2016-10-09 13:57:38 +03:00
|
|
|
|
|
|
|
def get_extension(media):
|
|
|
|
"""Gets the corresponding extension for any Telegram media"""
|
|
|
|
|
|
|
|
# Photos are always compressed as .jpg by Telegram
|
2017-10-13 12:38:12 +03:00
|
|
|
if isinstance(media, (UserProfilePhoto, ChatPhoto, MessageMediaPhoto)):
|
2016-10-09 13:57:38 +03:00
|
|
|
return '.jpg'
|
|
|
|
|
2017-08-24 18:44:38 +03:00
|
|
|
# Documents will come with a mime type
|
2016-10-09 13:57:38 +03:00
|
|
|
if isinstance(media, MessageMediaDocument):
|
2017-08-24 18:44:38 +03:00
|
|
|
if isinstance(media.document, Document):
|
|
|
|
if media.document.mime_type == 'application/octet-stream':
|
|
|
|
# Octet stream are just bytes, which have no default extension
|
|
|
|
return ''
|
|
|
|
else:
|
|
|
|
extension = guess_extension(media.document.mime_type)
|
|
|
|
return extension if extension else ''
|
|
|
|
|
|
|
|
return ''
|
2016-10-09 13:57:38 +03:00
|
|
|
|
|
|
|
|
2017-08-30 12:12:25 +03:00
|
|
|
def _raise_cast_fail(entity, target):
|
|
|
|
raise ValueError('Cannot cast {} to any kind of {}.'
|
|
|
|
.format(type(entity).__name__, target))
|
|
|
|
|
|
|
|
|
2017-10-05 14:14:54 +03:00
|
|
|
def get_input_peer(entity, allow_self=True):
|
2016-10-09 13:57:38 +03:00
|
|
|
"""Gets the input peer for the given "entity" (user, chat or channel).
|
2017-06-03 14:36:41 +03:00
|
|
|
A ValueError is raised if the given entity isn't a supported type."""
|
2017-08-30 12:12:25 +03:00
|
|
|
if not isinstance(entity, TLObject):
|
|
|
|
_raise_cast_fail(entity, 'InputPeer')
|
|
|
|
|
2017-09-29 14:11:33 +03:00
|
|
|
if type(entity).SUBCLASS_OF_ID == 0xc91c90b6: # crc32(b'InputPeer')
|
2017-01-17 22:22:47 +03:00
|
|
|
return entity
|
|
|
|
|
2016-10-09 13:57:38 +03:00
|
|
|
if isinstance(entity, User):
|
2017-10-05 14:14:54 +03:00
|
|
|
if entity.is_self and allow_self:
|
2017-07-10 17:09:20 +03:00
|
|
|
return InputPeerSelf()
|
|
|
|
else:
|
|
|
|
return InputPeerUser(entity.id, entity.access_hash)
|
2017-06-15 18:03:59 +03:00
|
|
|
|
2017-10-13 12:38:12 +03:00
|
|
|
if isinstance(entity, (Chat, ChatEmpty, ChatForbidden)):
|
2016-10-09 13:57:38 +03:00
|
|
|
return InputPeerChat(entity.id)
|
2017-06-15 18:03:59 +03:00
|
|
|
|
2017-10-13 12:38:12 +03:00
|
|
|
if isinstance(entity, (Channel, ChannelForbidden)):
|
2016-10-09 13:57:38 +03:00
|
|
|
return InputPeerChannel(entity.id, entity.access_hash)
|
|
|
|
|
2017-06-15 18:03:59 +03:00
|
|
|
# Less common cases
|
2017-07-10 17:09:20 +03:00
|
|
|
if isinstance(entity, UserEmpty):
|
|
|
|
return InputPeerEmpty()
|
|
|
|
|
|
|
|
if isinstance(entity, InputUser):
|
|
|
|
return InputPeerUser(entity.user_id, entity.access_hash)
|
|
|
|
|
2017-06-15 18:03:59 +03:00
|
|
|
if isinstance(entity, UserFull):
|
2017-07-10 17:09:20 +03:00
|
|
|
return get_input_peer(entity.user)
|
2017-06-15 18:03:59 +03:00
|
|
|
|
|
|
|
if isinstance(entity, ChatFull):
|
|
|
|
return InputPeerChat(entity.id)
|
|
|
|
|
2017-07-04 22:18:35 +03:00
|
|
|
if isinstance(entity, PeerChat):
|
|
|
|
return InputPeerChat(entity.chat_id)
|
|
|
|
|
2017-08-30 12:12:25 +03:00
|
|
|
_raise_cast_fail(entity, 'InputPeer')
|
2017-05-23 10:45:48 +03:00
|
|
|
|
2016-10-09 13:57:38 +03:00
|
|
|
|
2017-07-07 10:48:06 +03:00
|
|
|
def get_input_channel(entity):
|
2017-07-10 17:04:10 +03:00
|
|
|
"""Similar to get_input_peer, but for InputChannel's alone"""
|
2017-08-30 12:12:25 +03:00
|
|
|
if not isinstance(entity, TLObject):
|
|
|
|
_raise_cast_fail(entity, 'InputChannel')
|
|
|
|
|
2017-09-29 14:11:33 +03:00
|
|
|
if type(entity).SUBCLASS_OF_ID == 0x40f202fd: # crc32(b'InputChannel')
|
2017-07-07 10:48:06 +03:00
|
|
|
return entity
|
|
|
|
|
2017-10-13 12:38:12 +03:00
|
|
|
if isinstance(entity, (Channel, ChannelForbidden)):
|
2017-07-07 10:48:06 +03:00
|
|
|
return InputChannel(entity.id, entity.access_hash)
|
|
|
|
|
2017-08-05 10:37:34 +03:00
|
|
|
if isinstance(entity, InputPeerChannel):
|
|
|
|
return InputChannel(entity.channel_id, entity.access_hash)
|
|
|
|
|
2017-08-30 12:12:25 +03:00
|
|
|
_raise_cast_fail(entity, 'InputChannel')
|
2017-07-07 10:48:06 +03:00
|
|
|
|
|
|
|
|
2017-07-10 17:04:10 +03:00
|
|
|
def get_input_user(entity):
|
|
|
|
"""Similar to get_input_peer, but for InputUser's alone"""
|
2017-08-30 12:12:25 +03:00
|
|
|
if not isinstance(entity, TLObject):
|
|
|
|
_raise_cast_fail(entity, 'InputUser')
|
|
|
|
|
2017-09-29 14:11:33 +03:00
|
|
|
if type(entity).SUBCLASS_OF_ID == 0xe669bf46: # crc32(b'InputUser')
|
2017-07-10 17:04:10 +03:00
|
|
|
return entity
|
|
|
|
|
|
|
|
if isinstance(entity, User):
|
|
|
|
if entity.is_self:
|
|
|
|
return InputUserSelf()
|
|
|
|
else:
|
|
|
|
return InputUser(entity.id, entity.access_hash)
|
|
|
|
|
2017-10-24 10:42:51 +03:00
|
|
|
if isinstance(entity, InputPeerSelf):
|
|
|
|
return InputUserSelf()
|
|
|
|
|
|
|
|
if isinstance(entity, (UserEmpty, InputPeerEmpty)):
|
2017-07-10 17:04:10 +03:00
|
|
|
return InputUserEmpty()
|
|
|
|
|
|
|
|
if isinstance(entity, UserFull):
|
|
|
|
return get_input_user(entity.user)
|
|
|
|
|
2017-07-10 17:09:20 +03:00
|
|
|
if isinstance(entity, InputPeerUser):
|
|
|
|
return InputUser(entity.user_id, entity.access_hash)
|
|
|
|
|
2017-08-30 12:12:25 +03:00
|
|
|
_raise_cast_fail(entity, 'InputUser')
|
2017-07-10 17:04:10 +03:00
|
|
|
|
|
|
|
|
2017-09-25 14:43:03 +03:00
|
|
|
def get_input_document(document):
|
|
|
|
"""Similar to get_input_peer, but for documents"""
|
|
|
|
if not isinstance(document, TLObject):
|
|
|
|
_raise_cast_fail(document, 'InputDocument')
|
|
|
|
|
2017-09-29 14:11:33 +03:00
|
|
|
if type(document).SUBCLASS_OF_ID == 0xf33fdb68: # crc32(b'InputDocument')
|
2017-09-25 14:43:03 +03:00
|
|
|
return document
|
|
|
|
|
|
|
|
if isinstance(document, Document):
|
|
|
|
return InputDocument(id=document.id, access_hash=document.access_hash)
|
|
|
|
|
|
|
|
if isinstance(document, DocumentEmpty):
|
|
|
|
return InputDocumentEmpty()
|
|
|
|
|
|
|
|
if isinstance(document, MessageMediaDocument):
|
|
|
|
return get_input_document(document.document)
|
|
|
|
|
|
|
|
if isinstance(document, Message):
|
|
|
|
return get_input_document(document.media)
|
|
|
|
|
|
|
|
_raise_cast_fail(document, 'InputDocument')
|
|
|
|
|
|
|
|
|
|
|
|
def get_input_photo(photo):
|
|
|
|
"""Similar to get_input_peer, but for documents"""
|
|
|
|
if not isinstance(photo, TLObject):
|
|
|
|
_raise_cast_fail(photo, 'InputPhoto')
|
|
|
|
|
2017-09-29 14:11:33 +03:00
|
|
|
if type(photo).SUBCLASS_OF_ID == 0x846363e0: # crc32(b'InputPhoto')
|
2017-09-25 14:43:03 +03:00
|
|
|
return photo
|
|
|
|
|
2017-10-08 14:45:14 +03:00
|
|
|
if isinstance(photo, photos.Photo):
|
|
|
|
photo = photo.photo
|
|
|
|
|
2017-09-25 14:43:03 +03:00
|
|
|
if isinstance(photo, Photo):
|
|
|
|
return InputPhoto(id=photo.id, access_hash=photo.access_hash)
|
|
|
|
|
|
|
|
if isinstance(photo, PhotoEmpty):
|
|
|
|
return InputPhotoEmpty()
|
|
|
|
|
|
|
|
_raise_cast_fail(photo, 'InputPhoto')
|
|
|
|
|
|
|
|
|
|
|
|
def get_input_geo(geo):
|
|
|
|
"""Similar to get_input_peer, but for geo points"""
|
|
|
|
if not isinstance(geo, TLObject):
|
|
|
|
_raise_cast_fail(geo, 'InputGeoPoint')
|
|
|
|
|
2017-09-29 14:11:33 +03:00
|
|
|
if type(geo).SUBCLASS_OF_ID == 0x430d225: # crc32(b'InputGeoPoint')
|
2017-09-25 14:43:03 +03:00
|
|
|
return geo
|
|
|
|
|
|
|
|
if isinstance(geo, GeoPoint):
|
|
|
|
return InputGeoPoint(lat=geo.lat, long=geo.long)
|
|
|
|
|
|
|
|
if isinstance(geo, GeoPointEmpty):
|
|
|
|
return InputGeoPointEmpty()
|
|
|
|
|
|
|
|
if isinstance(geo, MessageMediaGeo):
|
|
|
|
return get_input_geo(geo.geo)
|
|
|
|
|
|
|
|
if isinstance(geo, Message):
|
|
|
|
return get_input_geo(geo.media)
|
|
|
|
|
|
|
|
_raise_cast_fail(geo, 'InputGeoPoint')
|
|
|
|
|
|
|
|
|
|
|
|
def get_input_media(media, user_caption=None, is_photo=False):
|
|
|
|
"""Similar to get_input_peer, but for media.
|
|
|
|
|
|
|
|
If the media is a file location and is_photo is known to be True,
|
|
|
|
it will be treated as an InputMediaUploadedPhoto.
|
|
|
|
"""
|
|
|
|
if not isinstance(media, TLObject):
|
|
|
|
_raise_cast_fail(media, 'InputMedia')
|
|
|
|
|
2017-09-29 14:11:33 +03:00
|
|
|
if type(media).SUBCLASS_OF_ID == 0xfaf846f4: # crc32(b'InputMedia')
|
2017-09-25 14:43:03 +03:00
|
|
|
return media
|
|
|
|
|
|
|
|
if isinstance(media, MessageMediaPhoto):
|
|
|
|
return InputMediaPhoto(
|
|
|
|
id=get_input_photo(media.photo),
|
|
|
|
caption=media.caption if user_caption is None else user_caption,
|
|
|
|
ttl_seconds=media.ttl_seconds
|
|
|
|
)
|
|
|
|
|
|
|
|
if isinstance(media, MessageMediaDocument):
|
|
|
|
return InputMediaDocument(
|
|
|
|
id=get_input_document(media.document),
|
|
|
|
caption=media.caption if user_caption is None else user_caption,
|
|
|
|
ttl_seconds=media.ttl_seconds
|
|
|
|
)
|
|
|
|
|
|
|
|
if isinstance(media, FileLocation):
|
|
|
|
if is_photo:
|
|
|
|
return InputMediaUploadedPhoto(
|
|
|
|
file=media,
|
|
|
|
caption=user_caption or ''
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
return InputMediaUploadedDocument(
|
|
|
|
file=media,
|
|
|
|
mime_type='application/octet-stream', # unknown, assume bytes
|
|
|
|
attributes=[DocumentAttributeFilename('unnamed')],
|
|
|
|
caption=user_caption or ''
|
|
|
|
)
|
|
|
|
|
|
|
|
if isinstance(media, MessageMediaGame):
|
|
|
|
return InputMediaGame(id=media.game.id)
|
|
|
|
|
2017-10-13 12:38:12 +03:00
|
|
|
if isinstance(media, (ChatPhoto, UserProfilePhoto)):
|
2017-09-25 14:43:03 +03:00
|
|
|
if isinstance(media.photo_big, FileLocationUnavailable):
|
|
|
|
return get_input_media(media.photo_small, is_photo=True)
|
|
|
|
else:
|
|
|
|
return get_input_media(media.photo_big, is_photo=True)
|
|
|
|
|
|
|
|
if isinstance(media, MessageMediaContact):
|
|
|
|
return InputMediaContact(
|
|
|
|
phone_number=media.phone_number,
|
|
|
|
first_name=media.first_name,
|
|
|
|
last_name=media.last_name
|
|
|
|
)
|
|
|
|
|
|
|
|
if isinstance(media, MessageMediaGeo):
|
|
|
|
return InputMediaGeoPoint(geo_point=get_input_geo(media.geo))
|
|
|
|
|
|
|
|
if isinstance(media, MessageMediaVenue):
|
|
|
|
return InputMediaVenue(
|
|
|
|
geo_point=get_input_geo(media.geo),
|
|
|
|
title=media.title,
|
|
|
|
address=media.address,
|
|
|
|
provider=media.provider,
|
|
|
|
venue_id=media.venue_id
|
|
|
|
)
|
|
|
|
|
2017-10-13 12:38:12 +03:00
|
|
|
if isinstance(media, (
|
2017-09-25 14:43:03 +03:00
|
|
|
MessageMediaEmpty, MessageMediaUnsupported,
|
2017-10-13 12:38:12 +03:00
|
|
|
ChatPhotoEmpty, UserProfilePhotoEmpty, FileLocationUnavailable)):
|
2017-09-25 14:43:03 +03:00
|
|
|
return InputMediaEmpty()
|
|
|
|
|
|
|
|
if isinstance(media, Message):
|
|
|
|
return get_input_media(media.media)
|
|
|
|
|
|
|
|
_raise_cast_fail(media, 'InputMedia')
|
|
|
|
|
|
|
|
|
2017-10-09 20:40:39 +03:00
|
|
|
def get_peer_id(peer, add_mark=False):
|
2017-10-01 14:24:04 +03:00
|
|
|
"""Finds the ID of the given peer, and optionally converts it to
|
|
|
|
the "bot api" format if 'add_mark' is set to True.
|
|
|
|
"""
|
2017-10-06 22:47:10 +03:00
|
|
|
# First we assert it's a Peer TLObject, or early return for integers
|
2017-10-05 13:59:44 +03:00
|
|
|
if not isinstance(peer, TLObject):
|
|
|
|
if isinstance(peer, int):
|
2017-10-09 20:40:39 +03:00
|
|
|
return peer
|
2017-10-05 13:59:44 +03:00
|
|
|
else:
|
|
|
|
_raise_cast_fail(peer, 'int')
|
|
|
|
|
2017-10-06 22:42:04 +03:00
|
|
|
elif type(peer).SUBCLASS_OF_ID not in {0x2d45687, 0xc91c90b6}:
|
2017-10-05 13:59:44 +03:00
|
|
|
# Not a Peer or an InputPeer, so first get its Input version
|
2017-10-05 14:14:54 +03:00
|
|
|
peer = get_input_peer(peer, allow_self=False)
|
2017-10-05 13:59:44 +03:00
|
|
|
|
2017-10-06 22:47:10 +03:00
|
|
|
# Set the right ID/kind, or raise if the TLObject is not recognised
|
2017-10-13 12:38:12 +03:00
|
|
|
if isinstance(peer, (PeerUser, InputPeerUser)):
|
2017-10-09 20:40:39 +03:00
|
|
|
return peer.user_id
|
2017-10-13 12:38:12 +03:00
|
|
|
elif isinstance(peer, (PeerChat, InputPeerChat)):
|
2017-10-09 20:40:39 +03:00
|
|
|
return -peer.chat_id if add_mark else peer.chat_id
|
2017-10-13 12:38:12 +03:00
|
|
|
elif isinstance(peer, (PeerChannel, InputPeerChannel)):
|
2017-10-09 20:40:39 +03:00
|
|
|
i = peer.channel_id
|
|
|
|
if add_mark:
|
2017-10-06 22:42:04 +03:00
|
|
|
# Concat -100 through math tricks, .to_supergroup() on Madeline
|
|
|
|
# IDs will be strictly positive -> log works
|
2017-10-09 20:40:39 +03:00
|
|
|
return -(i + pow(10, math.floor(math.log10(i) + 3)))
|
|
|
|
else:
|
|
|
|
return i
|
2017-10-06 22:42:04 +03:00
|
|
|
|
2017-10-09 20:40:39 +03:00
|
|
|
_raise_cast_fail(peer, 'int')
|
2017-10-01 14:24:04 +03:00
|
|
|
|
|
|
|
|
|
|
|
def resolve_id(marked_id):
|
|
|
|
"""Given a marked ID, returns the original ID and its Peer type"""
|
|
|
|
if marked_id >= 0:
|
|
|
|
return marked_id, PeerUser
|
|
|
|
|
|
|
|
if str(marked_id).startswith('-100'):
|
|
|
|
return int(str(marked_id)[4:]), PeerChannel
|
|
|
|
|
|
|
|
return -marked_id, PeerChat
|
|
|
|
|
|
|
|
|
2016-10-09 13:57:38 +03:00
|
|
|
def find_user_or_chat(peer, users, chats):
|
|
|
|
"""Finds the corresponding user or chat given a peer.
|
|
|
|
Returns None if it was not found"""
|
2017-10-01 11:50:37 +03:00
|
|
|
if isinstance(peer, PeerUser):
|
|
|
|
peer, where = peer.user_id, users
|
|
|
|
else:
|
|
|
|
where = chats
|
|
|
|
if isinstance(peer, PeerChat):
|
|
|
|
peer = peer.chat_id
|
2016-10-09 13:57:38 +03:00
|
|
|
elif isinstance(peer, PeerChannel):
|
2017-10-01 11:50:37 +03:00
|
|
|
peer = peer.channel_id
|
2017-06-14 15:06:35 +03:00
|
|
|
|
|
|
|
if isinstance(peer, int):
|
2017-10-01 11:50:37 +03:00
|
|
|
if isinstance(where, dict):
|
|
|
|
return where.get(peer)
|
|
|
|
else:
|
|
|
|
try:
|
|
|
|
return next(x for x in where if x.id == peer)
|
|
|
|
except StopIteration:
|
|
|
|
pass
|
2016-10-09 13:57:38 +03:00
|
|
|
|
|
|
|
|
2017-05-21 14:59:16 +03:00
|
|
|
def get_appropriated_part_size(file_size):
|
|
|
|
"""Gets the appropriated part size when uploading or downloading files,
|
2016-10-09 13:57:38 +03:00
|
|
|
given an initial file size"""
|
2017-10-09 14:19:03 +03:00
|
|
|
if file_size <= 104857600: # 100MB
|
2016-10-09 13:57:38 +03:00
|
|
|
return 128
|
|
|
|
if file_size <= 786432000: # 750MB
|
|
|
|
return 256
|
|
|
|
if file_size <= 1572864000: # 1500MB
|
|
|
|
return 512
|
|
|
|
|
|
|
|
raise ValueError('File size too large')
|