Source code for ehforwarderbot.message

# coding=utf-8

from abc import ABC, abstractmethod
from import Iterable as CIterable
from import Mapping as CMapping
from typing import IO, Dict, Optional, List, Any, Tuple, Iterable, Mapping

from .constants import *
from .chat import EFBChat
from .channel import EFBChannel
from . import coordinator

[docs]class EFBMsg: """A message. Attributes: attributes (Optional[:obj:`.EFBMsgAttribute`]): Attributes used for a specific message type. Only specific message type requires this attribute. Defaulted to ``None``. - Link: :obj:`.EFBMsgLinkAttribute` - Location: :obj:`.EFBMsgLocationAttribute` - Status: Typing/Sending files/etc.: :obj:`.EFBMsgStatusAttribute` .. Note:: Do NOT use object of the abstract class :class:`.EFBMsgAttribute` for ``attributes``, but object of specific class instead. author (:obj:`.EFBChat`): Author of this message. chat (:obj:`.EFBChat`): Sender of the message. commands (Optional[:obj:`EFBMsgCommands`]): Commands attached to the message deliver_to (:obj:`.EFBChannel`): The channel that the message is to be delivered to edit (bool): Flag this up if the message is edited. Flag only this if no multimedia file is modified, otherwise flag up both this one and ``edit_media`` as well. If no media file is modified, the edited message may carry no information about the file. edit_media (bool): Flag this up if any file attached to the message is modified. If this value is true, ``edit`` must also be ``True``. file (IO[bytes]): File object to multimedia file, type "ra". ``None`` if N/A. recommended to use ``NamedTemporaryFile`` object, the file can be deleted when closed, if not used otherwise. All file object must be rewind back to 0 (````) before sending. filename (str): File name of the multimedia file. ``None`` if N/A is_system (bool): Mark as true if this message is a system message. mime (str): MIME type of the file. ``None`` if N/A path (str): Local path of multimedia file. ``None`` if N/A reactions (Dict[str, Iterable[:obj:`EFBChat`]]): Indicate reactions to the message. Dictionary key represents the reaction name, usually an emoji. Value is a collection of users who reacted to the message with that certain emoji. All :obj:`EFBChat` objects in this dict must be of a user or a group member. substitutions (Optional[:obj:`EFBMsgSubstitutions`]): Substitutions of messages, usually used when the some parts of the text of the message refers to another user or chat. target (Optional[:obj:`EFBMsg`]): Target message (usually for messages that "replies to" another message). .. note:: This message may be a "minimum message", with only required fields: - :attr:`` - :attr:`` - :attr:`.EFBMsg.text` - :attr:`.EFBMsg.type` - :attr:`.EFBMsg.uid` text (str): text of the message type (:obj:`.MsgType`): Type of message uid (str): Unique ID of message. Usually stores the message ID from slave channel. This ID must be unique among all chats in the same channel. .. Note:: Some channels may not support message editing. Some channels may issue a new uid for edited message. vendor_specific (Dict[str, Any]): A series of vendor specific attributes attached. This can be used by any other channels or middlewares that is compatible with such information. Note that no guarantee is provided for information in this section. """ def __init__(self): self.attributes: Optional[EFBMsgAttribute] = None EFBChat = None EFBChat = None self.commands: Optional[EFBMsgCommands] = None self.deliver_to: EFBChannel = None self.edit: bool = False self.edit_media: bool = False self.file: Optional[IO[bytes]] = None self.filename: Optional[str] = None self.is_system: bool = False self.mime: Optional[str] = None self.path: Optional[str] = None self.reactions: Dict[str, Iterable[EFBChat]] = dict() self.substitutions: Optional[EFBMsgSubstitutions] = None Optional[EFBMsg] = None self.text: str = "" self.type: MsgType = MsgType.Unsupported self.uid: Optional[str] = None self.vendor_specific: Dict[str, Any] = dict() def __str__(self): return "<EFBMsg, {}@{} [{}]: {msg.text}; {msg.uid}>".format(msg=self) def __repr__(self): return "<EFBMsg, {}@{} [{}]: " \ "{msg.text}; " \ "Attributes: {msg.attributes}; " \ "Delivering to: {msg.deliver_to}; " \ "Edited: {msg.edit}; " \ "System message: {msg.is_system}; " \ "Substitutions: {msg.substitutions}; " \ "Target messages: {}; " \ "UID: {msg.uid}; " \ "Reactions: {msg.reactions}; " \ "File: {msg.file} ({msg.filename} @ {msg.path}), {msg.mime}; " \ "Vendor: {msg.vendor_specific}>".format(msg=self)
[docs] def verify(self): """ Verify the validity of message. """ if is None or not isinstance(, EFBChat): raise ValueError("Author is not valid.") else: if is None or not isinstance(, EFBChat): raise ValueError("Chat is not valid.") elif is not # Prevent repetitive verification if self.type is None or not isinstance(self.type, MsgType): raise ValueError("Type is not valid.") if self.deliver_to is None or not isinstance(self.deliver_to, EFBChannel): raise ValueError("Deliver_to is not valid.") if self.type in (MsgType.Audio, MsgType.File, MsgType.Image, MsgType.Sticker, MsgType.Video) and \ ((not self.edit) or (self.edit and self.edit_media)): if self.file is None or not hasattr(self.file, "read") or not hasattr(self.file, "close"): raise ValueError("File is not valid.") if self.mime is None or not self.mime or not isinstance(self.mime, str): raise ValueError("MIME is not valid.") if self.path is None or not self.path or not isinstance(self.path, str): raise ValueError("Path is not valid.") if self.type == MsgType.Location and (self.attributes is None or not isinstance(self.attributes, EFBMsgLocationAttribute)): raise ValueError("Attribute of location message is invalid.") if self.type == MsgType.Link and (self.attributes is None or not isinstance(self.attributes, EFBMsgLinkAttribute)): raise ValueError("Attribute of link message is invalid.") if self.type == MsgType.Status and (self.attributes is None or not isinstance(self.attributes, EFBMsgStatusAttribute)): raise ValueError("Attribute of status message is invalid.") if self.attributes: self.attributes.verify() if self.commands: self.commands.verify() if self.substitutions: self.substitutions.verify()
def __getstate__(self): state = self.__dict__.copy() del state['file'] if state['deliver_to'] is not None: state['deliver_to'] = state['deliver_to'].channel_id return state def __setstate__(self, state: Dict[str, Any]): self.__dict__.update(state) if self.path: self.file = open(self.path, 'rb') try: dt = coordinator.get_module_by_id(state['deliver_to']) if isinstance(dt, EFBChannel): self.deliver_to = dt except NameError: pass
[docs]class EFBMsgAttribute(ABC): """Abstract class of a message attribute.""" @abstractmethod def __init__(self): raise NotImplementedError("Do not use the abstract class EFBMsgAttribute") @abstractmethod def verify(self): raise NotImplementedError()
[docs]class EFBMsgLinkAttribute(EFBMsgAttribute): """ EFB link message attribute. Attributes: title (str): Title of the link. description (str, optional): Description of the link. image (str, optional): Image/thumbnail URL of the link. url (str): URL of the link. """ title: str = "" description: Optional[str] = None image: Optional[str] = None url: str = "" # noinspection PyMissingConstructor
[docs] def __init__(self, title: str = None, description: Optional[str] = None, image: Optional[str] = None, url: str = None): """ Args: title (str): Title of the link. description (str, optional): Description of the link. image (str, optional): Image/thumbnail URL of the link. url (str): URL of the link. """ if title is None or url is None: raise ValueError("Title and URL is required.") self.title = title self.description = description self.image = image self.url = url
def __str__(self): return "<EFBMsgLinkAttribute, {attr.title}: {attr.description} " \ "({attr.image}) @ {attr.url}>".format(attr=self) def verify(self): if self.url is None: raise ValueError("URL does not exist") if self.title is None: raise ValueError("Title does not exist")
[docs]class EFBMsgLocationAttribute(EFBMsgAttribute): """ EFB location message attribute. Attributes: latitude (float): Latitude of the location. longitude (float): Longitude of the location. """ latitude: float = 0 longitude: float = 0 # noinspection PyMissingConstructor
[docs] def __init__(self, latitude: float, longitude: float): """ Args: latitude (float): Latitude of the location. longitude (float): Longitude of the location. """ self.latitude = latitude self.longitude = longitude
def __str__(self): return "<EFBMsgLocationAttribute: {attr.latitude}, {attr.longitude}>".format(attr=self) def verify(self): if self.latitude is None or not isinstance(self.latitude, float): raise ValueError("Latitude is invalid.") if self.longitude is None or not isinstance(self.longitude, float): raise ValueError("Longitude is invalid.")
[docs]class EFBMsgCommand: """ EFB message command. This object records a way to call a method in the module object. In case where the message has an ``author`` from a different module from the ``chat``, this function should be called on the ``author``'s module. Attributes: name (str): Human-friendly name of the command. callable_name (str): Callable name of the command. args (Iterable[Any]): Arguments passed to the function. kwargs (Mapping[str, Any]): Keyword arguments passed to the function. """ name: str = "" callable_name: str = "" args: Tuple = tuple() kwargs: Mapping[str, Any] = {}
[docs] def __init__(self, name: str, callable_name: str, args: Iterable[Any] = None, kwargs: Optional[Mapping[str, Any]] = None): """ Args: name (str): Human-friendly name of the command. callable_name (str): Callable name of the command. args (Optional[Iterable[Any]]): Arguments passed to the function. Defaulted to empty list; kwargs (Optional[Mapping[str, Any]]): Keyword arguments passed to the function. Defaulted to empty dict. """ if args is None: args = tuple() if kwargs is None: kwargs = dict() if not isinstance(name, str): raise TypeError("name must be a string.") if not isinstance(callable_name, str): raise TypeError("callable must be a string.") if not isinstance(args, CIterable): raise TypeError("args must be an iterable.") if not isinstance(kwargs, CMapping): raise TypeError("kwargs must be a mapping.") = name self.callable_name = callable_name self.args = tuple(args) self.kwargs = dict(kwargs)
def __str__(self): return "<EFBMsgCommand: {name}, {callable_name}({params})>".format(, callable_name=self.callable_name, params=", ".join(self.args + tuple("%r=%r" % i for i in self.kwargs.items())) ) def verify(self): if not raise ValueError("Name does not exist.") if not self.callable_name: raise ValueError("Callable does not exist.")
[docs]class EFBMsgCommands: """ EFB message commands. Message commands allow user to take action to a specific message, including vote, add friends, etc. Attributes: commands (list of :obj:`EFBMsgCommand`): Commands for the message. """ commands: List[EFBMsgCommand] = []
[docs] def __init__(self, commands: List[EFBMsgCommand]): """ Args: commands (list of :obj:`EFBMsgCommand`): Commands for the message. """ if not (isinstance(commands, list) and len(commands) > 0 and all( isinstance(i, EFBMsgCommand) for i in commands)): raise ValueError("There must be one or more commands, " "and all of them must be in type EFBMsgCommand.") self.commands = commands.copy()
def __str__(self): return str(self.commands) def verify(self): for i in self.commands: i.verify()
[docs]class EFBMsgStatusAttribute(EFBMsgAttribute): """ EFB Message status attribute. Message with type ``Status`` notifies the other end to update a chat-specific status, such as typing, send files, etc. Attributes: status_type: Type of status, possible values are defined in the ``EFBMsgStatusAttribute``. timeout (Optional[int]): Number of milliseconds for this status to expire. Default to 5 seconds. Types: List of status types supported """
[docs] class Types(Enum): """ Attributes: TYPING: Used in :attr:`~.ehforwarderbot.message.EFBMsgStatusAttribute.status_type`, represent the status of typing. UPLOADING_FILE: Used in :attr:`~.ehforwarderbot.message.EFBMsgStatusAttribute.status_type`, represent the status of uploading file. UPLOADING_IMAGE: Used in :attr:`~.ehforwarderbot.message.EFBMsgStatusAttribute.status_type`, represent the status of uploading image. UPLOADING_AUDIO: Used in :attr:`~.ehforwarderbot.message.EFBMsgStatusAttribute.status_type`, represent the status of uploading audio. UPLOADING_VIDEO: Used in :attr:`~.ehforwarderbot.message.EFBMsgStatusAttribute.status_type`, represent the status of uploading video. """ TYPING = "TYPING" UPLOADING_FILE = "UPLOADING_FILE" UPLOADING_IMAGE = "UPLOADING_IMAGE" UPLOADING_AUDIO = "UPLOADING_AUDIO" UPLOADING_VIDEO = "UPLOADING_VIDEO"
# noinspection PyMissingConstructor
[docs] def __init__(self, status_type: Types, timeout: int = 5000): """ Args: status_type: Type of status. timeout (Optional[int]): Number of milliseconds for this status to expire. Default to 5 seconds. """ self.status_type: 'EFBMsgStatusAttribute.Types' = status_type self.timeout: int = timeout
def __str__(self): return "<EFBMsgStatusAttribute: {attr.status_type} @ {attr.timeout}ms>".format(attr=self) def verify(self): if self.status_type is None or not isinstance(self.status_type, self.Types): raise ValueError("Status type is invalid.") if not isinstance(self.timeout, int): raise ValueError("Timeout is invalid.")
[docs]class EFBMsgSubstitutions(dict): """ EFB message substitutions. This is for the case when user "@-referred" a list of users in the message. Substitutions here is a dict of correspondence between the string used to refer to the user in the message and a user object. Dictionary of text substitutions targeting to a user or member. The key of the dictionary is a tuple of two :obj:`int`\\ s, where first of it is the starting position in the string, and the second is the ending position defined similar to Python's substring. A tuple of ``(3, 15)`` corresponds to ``msg.text[3:15]``. The value of the tuple ``(a, b)`` must lie within ``a ∈ [0, l)``, ``b ∈ (a, l]``, where ``l`` is the length of the message text. Value of the dict may be any user of the chat, or a member of a group. Notice that the :obj:`EFBChat` object here must NOT be a group. Type: Dict[Tuple[int, int], :obj:`.EFBChat`] """ def __init__(self, substitutions: Dict[Tuple[int, int], EFBChat]): if not isinstance(substitutions, dict): raise TypeError("Substitutions must be a dict.") for i in substitutions: if not isinstance(i, tuple) or not len(i) == 2 or not isinstance(i[0], int) or not isinstance(i[1], int) \ or not i[0] < i[1]: raise TypeError("Index of substitution {} must be a tuple of 2 integers where the first one is less" "than the second one.".format(i)) if not isinstance(substitutions[i], EFBChat): raise TypeError("Substitution {} is not a chat object.".format(i)) if substitutions[i].is_chat and \ substitutions[i].chat_type == ChatType.Group: raise ValueError("Substitution {} is a group.".format(i)) super().__init__(substitutions) def verify(self): ranges = sorted(self.keys()) if ranges and (ranges[0][0] < 0 or ranges[0][1] < ranges[0][0]): raise ValueError("Index %s is invalid." % ranges[0]) for i in range(1, len(ranges)): if ranges[i][0] < 0 or ranges[i][1] < ranges[i][0]: raise ValueError("Index %s is invalid." % ranges[i]) if ranges[i][0] < ranges[i - 1][1]: raise ValueError("Index %s overlaps with %s." % (ranges[i], ranges[i - 1])) for i in self.values(): i.verify()