From 32f3ce596a0c8f9c8e82f124171b76296bddcb0f Mon Sep 17 00:00:00 2001 From: Snipy7374 <100313469+Snipy7374@users.noreply.github.com> Date: Wed, 11 Oct 2023 23:20:22 +0200 Subject: [PATCH 1/2] feat: add new properties to disnake.Attachment --- changelog/1118.feature.rst | 1 + disnake/message.py | 19 +++++++++++++++++++ 2 files changed, 20 insertions(+) create mode 100644 changelog/1118.feature.rst diff --git a/changelog/1118.feature.rst b/changelog/1118.feature.rst new file mode 100644 index 0000000000..ce4dcb3a05 --- /dev/null +++ b/changelog/1118.feature.rst @@ -0,0 +1 @@ +Add :attr:`.Attachment.expires_at` and :attr:`.Attachment.is_expired` properties to :class:`.Attachment`. diff --git a/disnake/message.py b/disnake/message.py index 21f59e269e..ebb470f76e 100644 --- a/disnake/message.py +++ b/disnake/message.py @@ -21,6 +21,7 @@ cast, overload, ) +from urllib.parse import parse_qs, urlparse from . import utils from .components import ActionRow, MessageComponent, _component_factory @@ -302,6 +303,7 @@ class Attachment(Hashable): "description", "duration", "waveform", + "_ex", "_flags", ) @@ -321,8 +323,25 @@ def __init__(self, *, data: AttachmentPayload, state: ConnectionState) -> None: self.waveform: Optional[bytes] = ( b64decode(waveform_data) if (waveform_data := data.get("waveform")) else None ) + _params = urlparse(self.url) + self._ex = parse_qs(_params.query)["ex"][0] self._flags: int = data.get("flags", 0) + def expires_at(self) -> datetime.datetime: + """The date when this attachment will expire. + + :return type: :class:`datetime.datetime` + """ + timestamp = int(self._ex, 16) + return datetime.datetime.fromtimestamp(timestamp, tz=datetime.timezone.utc) + + def is_expired(self) -> bool: + """Whether this attachment expired or not. + + :return type: :class:`bool` + """ + return utils.utcnow() >= self.expires_at() + def is_spoiler(self) -> bool: """Whether this attachment contains a spoiler. From 424716036b865c691b722e103e3c8957f0e398c6 Mon Sep 17 00:00:00 2001 From: Snipy7374 <100313469+Snipy7374@users.noreply.github.com> Date: Thu, 12 Oct 2023 08:11:02 +0200 Subject: [PATCH 2/2] handle cases where attachments doesn't have new params --- disnake/message.py | 18 +++++++++++++----- 1 file changed, 13 insertions(+), 5 deletions(-) diff --git a/disnake/message.py b/disnake/message.py index ebb470f76e..5620a1b349 100644 --- a/disnake/message.py +++ b/disnake/message.py @@ -324,15 +324,19 @@ def __init__(self, *, data: AttachmentPayload, state: ConnectionState) -> None: b64decode(waveform_data) if (waveform_data := data.get("waveform")) else None ) _params = urlparse(self.url) - self._ex = parse_qs(_params.query)["ex"][0] + self._ex = parse_qs(_params.query).get("ex") self._flags: int = data.get("flags", 0) - def expires_at(self) -> datetime.datetime: + def expires_at(self) -> Optional[datetime.datetime]: """The date when this attachment will expire. + ``None`` if the ``ex`` param is not present in :attr:`.Attachment.url`. - :return type: :class:`datetime.datetime` + :return type: Optional[:class:`datetime.datetime`] """ - timestamp = int(self._ex, 16) + if not self._ex: + return + + timestamp = int(self._ex[0], 16) return datetime.datetime.fromtimestamp(timestamp, tz=datetime.timezone.utc) def is_expired(self) -> bool: @@ -340,7 +344,11 @@ def is_expired(self) -> bool: :return type: :class:`bool` """ - return utils.utcnow() >= self.expires_at() + ex = self.expires_at() + if not ex: + return False + + return utils.utcnow() >= ex def is_spoiler(self) -> bool: """Whether this attachment contains a spoiler.