diff --git a/discord/enums.py b/discord/enums.py index 4698de40d..20d799f04 100644 --- a/discord/enums.py +++ b/discord/enums.py @@ -232,6 +232,7 @@ class MessageType(Enum): guild_invite_reminder = 22 context_menu_command = 23 auto_moderation_action = 24 + role_subscription_purchase = 25 class SpeakingState(Enum): diff --git a/discord/flags.py b/discord/flags.py index a3b9eb5f2..a886ca735 100644 --- a/discord/flags.py +++ b/discord/flags.py @@ -290,6 +290,24 @@ class SystemChannelFlags(BaseFlags): """ return 8 + @flag_value + def role_subscription_purchase_notifications(self): + """:class:`bool`: Returns ``True`` if role subscription purchase and renewal + notifications are enabled. + + .. versionadded:: 2.2 + """ + return 16 + + @flag_value + def role_subscription_purchase_notification_replies(self): + """:class:`bool`: Returns ``True`` if the role subscription notifications + have a sticker reply button. + + .. versionadded:: 2.2 + """ + return 32 + @fill_with_flags() class MessageFlags(BaseFlags): diff --git a/discord/message.py b/discord/message.py index ea27ca96c..b959c8c42 100644 --- a/discord/message.py +++ b/discord/message.py @@ -73,6 +73,7 @@ if TYPE_CHECKING: MessageReference as MessageReferencePayload, MessageApplication as MessageApplicationPayload, MessageActivity as MessageActivityPayload, + RoleSubscriptionData as RoleSubscriptionDataPayload, ) from .types.interactions import MessageInteraction as MessageInteractionPayload @@ -108,6 +109,7 @@ __all__ = ( 'MessageReference', 'DeletedReferencedMessage', 'MessageApplication', + 'RoleSubscriptionInfo', ) @@ -657,6 +659,39 @@ class MessageApplication: return None +class RoleSubscriptionInfo: + """Represents a message's role subscription information. + + This is currently only attached to messages of type :attr:`MessageType.role_subscription_purchase`. + + .. versionadded:: 2.0 + + Attributes + ----------- + role_subscription_listing_id: :class:`int` + The ID of the SKU and listing that the user is subscribed to. + tier_name: :class:`str` + The name of the tier that the user is subscribed to. + total_months_subscribed: :class:`int` + The cumulative number of months that the user has been subscribed for. + is_renewal: :class:`bool` + Whether this notification is for a renewal rather than a new purchase. + """ + + __slots__ = ( + 'role_subscription_listing_id', + 'tier_name', + 'total_months_subscribed', + 'is_renewal', + ) + + def __init__(self, data: RoleSubscriptionDataPayload) -> None: + self.role_subscription_listing_id: int = int(data['role_subscription_listing_id']) + self.tier_name: str = data['tier_name'] + self.total_months_subscribed: int = data['total_months_subscribed'] + self.is_renewal: bool = data['is_renewal'] + + class PartialMessage(Hashable): """Represents a partial message to aid with working messages when only a message and channel ID are present. @@ -1399,6 +1434,11 @@ class Message(PartialMessage, Hashable): The interaction that this message is a response to. .. versionadded:: 2.0 + role_subscription: Optional[:class:`RoleSubscriptionInfo`] + The data of the role subscription purchase or renewal that prompted this + :attr:`MessageType.role_subscription_purchase` message. + + .. versionadded:: 2.2 guild: Optional[:class:`Guild`] The guild that the message belongs to, if applicable. """ @@ -1431,6 +1471,7 @@ class Message(PartialMessage, Hashable): 'stickers', 'components', 'interaction', + 'role_subscription', ) if TYPE_CHECKING: @@ -1516,6 +1557,14 @@ class Message(PartialMessage, Hashable): else: self.application = MessageApplication(state=self._state, data=application) + self.role_subscription: Optional[RoleSubscriptionInfo] = None + try: + role_subscription = data['role_subscription_data'] + except KeyError: + pass + else: + self.role_subscription = RoleSubscriptionInfo(role_subscription) + for handler in ('author', 'member', 'mentions', 'mention_roles', 'components'): try: getattr(self, f'_handle_{handler}')(data[handler]) @@ -1939,6 +1988,12 @@ class Message(PartialMessage, Hashable): if self.type is MessageType.guild_invite_reminder: return 'Wondering who to invite?\nStart by inviting anyone who can help you build the server!' + if self.type is MessageType.role_subscription_purchase and self.role_subscription is not None: + # TODO: figure out how the message looks like for is_renewal: true + total_months = self.role_subscription.total_months_subscribed + months = '1 month' if total_months == 1 else f'{total_months} months' + return f'{self.author.name} joined {self.role_subscription.tier_name} and has been a subscriber of {self.guild} for {months}!' + # Fallback for unknown message types return '' diff --git a/discord/role.py b/discord/role.py index 9d4bfa800..f2a3c2c4d 100644 --- a/discord/role.py +++ b/discord/role.py @@ -65,22 +65,31 @@ class RoleTags: The bot's user ID that manages this role. integration_id: Optional[:class:`int`] The integration ID that manages the role. + subscription_listing_id: Optional[:class:`int`] + The ID of this role's subscription SKU and listing. + + .. versionadded:: 2.2 """ __slots__ = ( 'bot_id', 'integration_id', '_premium_subscriber', + '_available_for_purchase', + 'subscription_listing_id', ) def __init__(self, data: RoleTagPayload): self.bot_id: Optional[int] = _get_as_snowflake(data, 'bot_id') self.integration_id: Optional[int] = _get_as_snowflake(data, 'integration_id') + self.subscription_listing_id: Optional[int] = _get_as_snowflake(data, 'subscription_listing_id') + # NOTE: The API returns "null" for this if it's valid, which corresponds to None. # This is different from other fields where "null" means "not there". # So in this case, a value of None is the same as True. # Which means we would need a different sentinel. self._premium_subscriber: Optional[Any] = data.get('premium_subscriber', MISSING) + self._available_for_purchase: Optional[Any] = data.get('available_for_purchase', MISSING) def is_bot_managed(self) -> bool: """:class:`bool`: Whether the role is associated with a bot.""" @@ -94,6 +103,13 @@ class RoleTags: """:class:`bool`: Whether the role is managed by an integration.""" return self.integration_id is not None + def is_available_for_purchase(self) -> bool: + """:class:`bool`: Whether the role is available for purchase. + + .. versionadded:: 2.2 + """ + return self._available_for_purchase is None + def __repr__(self) -> str: return ( f'