From 1767be0081070501b2a4e5d1a55b4386ea24bc89 Mon Sep 17 00:00:00 2001 From: Andrin S <65789180+Puncher1@users.noreply.github.com> Date: Thu, 20 Apr 2023 06:08:04 +0200 Subject: [PATCH] Add support for voice messages Co-authored-by: Danny <1695103+Rapptz@users.noreply.github.com> --- discord/flags.py | 8 ++++++++ discord/message.py | 18 ++++++++++++++++++ discord/permissions.py | 16 ++++++++++++++-- discord/types/message.py | 2 ++ discord/utils.py | 6 +++++- 5 files changed, 47 insertions(+), 3 deletions(-) diff --git a/discord/flags.py b/discord/flags.py index d0938583c..837aefc42 100644 --- a/discord/flags.py +++ b/discord/flags.py @@ -451,6 +451,14 @@ class MessageFlags(BaseFlags): """ return 4096 + @flag_value + def voice(self): + """:class:`bool`: Returns ``True`` if the message is a voice message. + + .. versionadded:: 2.3 + """ + return 8192 + @fill_with_flags() class PublicUserFlags(BaseFlags): diff --git a/discord/message.py b/discord/message.py index c96c7e022..f99fa01fb 100644 --- a/discord/message.py +++ b/discord/message.py @@ -183,6 +183,14 @@ class Attachment(Hashable): Whether the attachment is ephemeral. .. versionadded:: 2.0 + duration: Optional[:class:`float`] + The duration of the audio file in seconds. Returns ``None`` if it's not a voice message. + + .. versionadded:: 2.3 + waveform: Optional[:class:`bytes`] + The waveform (amplitudes) of the audio in bytes. Returns ``None`` if it's not a voice message. + + .. versionadded:: 2.3 """ __slots__ = ( @@ -197,6 +205,8 @@ class Attachment(Hashable): 'content_type', 'description', 'ephemeral', + 'duration', + 'waveform', ) def __init__(self, *, data: AttachmentPayload, state: ConnectionState): @@ -211,11 +221,19 @@ class Attachment(Hashable): self.content_type: Optional[str] = data.get('content_type') self.description: Optional[str] = data.get('description') self.ephemeral: bool = data.get('ephemeral', False) + self.duration: Optional[float] = data.get('duration_secs') + + waveform = data.get('waveform') + self.waveform: Optional[bytes] = utils._base64_to_bytes(waveform) if waveform is not None else None def is_spoiler(self) -> bool: """:class:`bool`: Whether this attachment contains a spoiler.""" return self.filename.startswith('SPOILER_') + def is_voice_message(self) -> bool: + """:class:`bool`: Whether this attachment is a voice message.""" + return self.duration is not None and 'voice-message' in self.url + def __repr__(self) -> str: return f'' diff --git a/discord/permissions.py b/discord/permissions.py index 9716f07c6..6be0674c9 100644 --- a/discord/permissions.py +++ b/discord/permissions.py @@ -177,7 +177,7 @@ class Permissions(BaseFlags): """A factory method that creates a :class:`Permissions` with all permissions set to ``True``. """ - return cls(0b1111111111111111111111111111111111111111111111) + return cls(0b11111111111111111111111111111111111111111111111) @classmethod def _timeout_mask(cls) -> int: @@ -261,8 +261,11 @@ class Permissions(BaseFlags): .. versionchanged:: 2.0 Added :attr:`create_public_threads`, :attr:`create_private_threads`, :attr:`manage_threads`, :attr:`send_messages_in_threads` and :attr:`use_external_stickers` permissions. + + .. versionchanged:: 2.3 + Added :attr:`send_voice_messages` permission. """ - return cls(0b111110010000000000001111111100001000000) + return cls(0b10000000111110010000000000001111111100001000000) @classmethod def voice(cls) -> Self: @@ -671,6 +674,14 @@ class Permissions(BaseFlags): """ return 1 << 45 + @flag_value + def send_voice_messages(self) -> int: + """:class:`bool`: Returns ``True`` if a user can send voice messages. + + .. versionadded:: 2.3 + """ + return 1 << 46 + def _augment_from_permissions(cls): cls.VALID_NAMES = set(Permissions.VALID_FLAGS) @@ -788,6 +799,7 @@ class PermissionOverwrite: moderate_members: Optional[bool] use_soundboard: Optional[bool] use_external_sounds: Optional[bool] + send_voice_messages: Optional[bool] def __init__(self, **kwargs: Optional[bool]): self._values: Dict[str, Optional[bool]] = {} diff --git a/discord/types/message.py b/discord/types/message.py index 1319b2e65..1157100fc 100644 --- a/discord/types/message.py +++ b/discord/types/message.py @@ -68,6 +68,8 @@ class Attachment(TypedDict): content_type: NotRequired[str] spoiler: NotRequired[bool] ephemeral: NotRequired[bool] + duration_secs: NotRequired[float] + waveform: NotRequired[str] MessageActivityType = Literal[1, 2, 3, 5] diff --git a/discord/utils.py b/discord/utils.py index fd711387b..9fe664caf 100644 --- a/discord/utils.py +++ b/discord/utils.py @@ -56,7 +56,7 @@ from typing import ( TYPE_CHECKING, ) import unicodedata -from base64 import b64encode +from base64 import b64encode, b64decode from bisect import bisect_left import datetime import functools @@ -628,6 +628,10 @@ def _bytes_to_base64_data(data: bytes) -> str: return fmt.format(mime=mime, data=b64) +def _base64_to_bytes(data: str) -> bytes: + return b64decode(data.encode('ascii')) + + def _is_submodule(parent: str, child: str) -> bool: return parent == child or child.startswith(parent + '.')