# Pyrofork - Telegram MTProto API Client Library for Python # Copyright (C) 2022-present Mayuri-Chan # # This file is part of Pyrofork. # # Pyrofork is free software: you can redistribute it and/or modify # it under the terms of the GNU Lesser General Public License as published # by the Free Software Foundation, either version 3 of the License, or # (at your option) any later version. # # Pyrofork 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 General Public License for more details. # # You should have received a copy of the GNU Lesser General Public License # along with Pyrofork. If not, see . import os import re from typing import BinaryIO, List, Union import pyrogram from pyrogram import enums, raw, types, utils from pyrogram.file_id import FileType class SendStory: def _split(self, message, entities, *args, **kwargs): return message, entities async def _upload_video( self: "pyrogram.Client", file_name: str, video: Union[str, BinaryIO] ): file = await self.save_file(video) return raw.types.InputMediaUploadedDocument( mime_type=self.guess_mime_type(file_name or video.name) or "video/mp4", file=file, attributes=[ raw.types.DocumentAttributeVideo( supports_streaming=True, duration=0, w=0, h=0 ) ] ) async def send_story( self: "pyrogram.Client", chat_id: Union[int,str] = None, privacy: "enums.StoriesPrivacyRules" = None, allowed_users: List[int] = None, denied_users: List[int] = None, #allowed_chats: List[int] = None, #denied_chats: List[int] = None, photo: Union[str, BinaryIO] = None, video: Union[str, BinaryIO] = None, file_name: str = None, pinned: bool = None, protect_content: bool = None, caption: str = None, parse_mode: "enums.ParseMode" = None, caption_entities: List["types.MessageEntity"] = None, period: int = None, forward_from_chat_id: Union[int, str] = None, forward_from_story_id: int = None, media_areas: List["types.InputMediaArea"] = None ) -> "types.Story": """Send new story. .. include:: /_includes/usable-by/users.rst Note: You must pass one of following paramater *animation*, *photo*, *video* Parameters: chat_id (``int`` | ``str``, *optional*): Unique identifier (int) or username (str) of the target channel. You can also use channel public link in form of *t.me/* (str). photo (``str`` | ``BinaryIO``, *optional*): Photo to send. Pass a file_id as string to send a photo that exists on the Telegram servers, pass an HTTP URL as a string for Telegram to get a photo from the Internet, pass a file path as string to upload a new photo that exists on your local machine, or pass a binary file-like object with its attribute ".name" set for in-memory uploads. video (``str`` | ``BinaryIO``, *optional*): Video to send. Pass a file_id as string to send a video that exists on the Telegram servers, pass an HTTP URL as a string for Telegram to get a video from the Internet, pass a file path as string to upload a new video that exists on your local machine, or pass a binary file-like object with its attribute ".name" set for in-memory uploads. file_name (``str``, *optional*): File name of the video sent. Defaults to file's path basename. privacy (:obj:`~pyrogram.enums.StoriesPrivacyRules`, *optional*): Story privacy. Defaults to :obj:`~pyrogram.enums.StoriesPrivacyRules.PUBLIC` allowed_users (List of ``int``, *optional*): List of user_id whos allowed to view the story. denied_users (List of ``int``, *optional*): List of user_id whos denied to view the story. pinned (``bool``, *optional*): if True, the story will be pinned. default to False. protect_content (``bool``, *optional*): Protects the contents of the sent story from forwarding and saving. default to False. caption (``str``, *optional*): Story caption, 0-1024 characters. parse_mode (:obj:`~pyrogram.enums.ParseMode`, *optional*): By default, texts are parsed using both Markdown and HTML styles. You can combine both syntaxes together. caption_entities (List of :obj:`~pyrogram.types.MessageEntity`): List of special entities that appear in the caption, which can be specified instead of *parse_mode*. period (``int``, *optional*): How long the story will posted, in secs. only for premium users. media_areas (List of :obj:`~pyrogram.types.InputMediaArea`): List of media area object to be included in story. forward_from_chat_id (``int`` | ``str``, *optional): Unique identifier (int) or username (str) of the source chat where the original story was sent. For your personal story you can simply use "me" or "self". For a contact that exists in your Telegram address book you can use his phone number (str). forward_from_story_id (``int``, *optional*): Single story id. Returns: :obj:`~pyrogram.types.Story` a single story is returned. Example: .. code-block:: python # Send new photo story photo_id = "abcd12345" await app.send_story(photo=photo_id, caption='Hello guys.') Raises: ValueError: In case of invalid arguments. """ if chat_id: peer = await self.resolve_peer(chat_id) else: peer = await self.resolve_peer("me") if privacy: privacy_rules = [types.StoriesPrivacyRules(type=privacy)] else: privacy_rules = [types.StoriesPrivacyRules(type=enums.StoriesPrivacyRules.PUBLIC)] if photo: if isinstance(photo, str): if os.path.isfile(photo): file = await self.save_file(photo) media = raw.types.InputMediaUploadedPhoto( file=file ) elif re.match("^https?://", photo): media = raw.types.InputMediaPhotoExternal( url=photo ) else: media = utils.get_input_media_from_file_id(photo, FileType.PHOTO) else: file = await self.save_file(photo) media = raw.types.InputMediaUploadedPhoto( file=file ) elif video: if isinstance(video, str): if os.path.isfile(video): file = await self.save_file(video) media = raw.types.InputMediaUploadedDocument( mime_type=self.guess_mime_type(video) or "video/mp4", file=file, attributes=[ raw.types.DocumentAttributeVideo( supports_streaming=True, duration=0, w=0, h=0 ) ] ) elif re.match("^https?://", video): media = raw.types.InputMediaDocumentExternal( url=video ) else: video = await self.download_media(video, in_memory=True) media = await self._upload_video(file_name,video) else: media = await self._upload_video(file_name,video) else: if forward_from_chat_id is None: raise ValueError("You need to pass one of the following parameter photo/video/forward_from_chat_id!") text, entities = self._split(**await utils.parse_text_entities(self, caption, parse_mode, caption_entities)) #if allowed_chats and len(allowed_chats) > 0: # chats = [await self.resolve_peer(chat_id) for chat_id in allowed_chats] # privacy_rules.append(raw.types.InputPrivacyValueAllowChatParticipants(chats=chats)) #if denied_chats and len(denied_chats) > 0: # chats = [await self.resolve_peer(chat_id) for chat_id in denied_chats] # privacy_rules.append(raw.types.InputPrivacyValueDisallowChatParticipants(chats=chats)) if allowed_users and len(allowed_users) > 0: users = [await self.resolve_peer(user_id) for user_id in allowed_users] privacy_rules.append(raw.types.InputPrivacyValueAllowUsers(users=users)) if denied_users and len(denied_users) > 0: users = [await self.resolve_peer(user_id) for user_id in denied_users] privacy_rules.append(raw.types.InputPrivacyValueDisallowUsers(users=users)) forward_from_chat = None if forward_from_chat_id is not None: forward_from_chat = await self.resolve_peer(forward_from_chat_id) media = raw.types.InputMediaEmpty() if forward_from_story_id is None: raise ValueError("You need to pass forward_from_story_id to forward story!") r = await self.invoke( raw.functions.stories.SendStory( peer=peer, media=media, privacy_rules=privacy_rules, random_id=self.rnd_id(), pinned=pinned, noforwards=protect_content, caption=text, entities=entities, period=period, fwd_from_id=forward_from_chat, fwd_from_story=forward_from_story_id if forward_from_chat_id is not None else None, fwd_modified=True if forward_from_chat_id is not None and caption is not None else False, media_areas=[ await media_area.write(self) for media_area in media_areas ] if media_areas is not None else None ) ) return await types.Story._parse(self, r.updates[0].story, r.updates[0].peer)