stormbrigade_sheriff/sbsheriff/Lib/site-packages/disnake/guild_scheduled_event.py

472 lines
17 KiB
Python

"""
The MIT License (MIT)
Copyright (c) 2021-present Disnake Development
Permission is hereby granted, free of charge, to any person obtaining a
copy of this software and associated documentation files (the "Software"),
to deal in the Software without restriction, including without limitation
the rights to use, copy, modify, merge, publish, distribute, sublicense,
and/or sell copies of the Software, and to permit persons to whom the
Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
DEALINGS IN THE SOFTWARE.
"""
from __future__ import annotations
from datetime import datetime
from typing import TYPE_CHECKING, Any, Dict, Optional
from .asset import Asset
from .enums import (
GuildScheduledEventEntityType,
GuildScheduledEventPrivacyLevel,
GuildScheduledEventStatus,
try_enum,
)
from .iterators import GuildScheduledEventUserIterator
from .mixins import Hashable
from .user import User
from .utils import (
MISSING,
_assetbytes_to_base64_data,
_get_as_snowflake,
cached_slot_property,
parse_time,
)
if TYPE_CHECKING:
from .abc import GuildChannel, Snowflake
from .asset import AssetBytes
from .guild import Guild
from .state import ConnectionState
from .types.guild_scheduled_event import (
GuildScheduledEvent as GuildScheduledEventPayload,
GuildScheduledEventEntityMetadata as GuildScheduledEventEntityMetadataPayload,
)
__all__ = ("GuildScheduledEventMetadata", "GuildScheduledEvent")
class GuildScheduledEventMetadata:
"""
Represents a guild scheduled event entity metadata.
.. versionadded:: 2.3
Attributes
----------
location: Optional[:class:`str`]
The location of the guild scheduled event. If :attr:`GuildScheduledEvent.entity_type` is
:class:`GuildScheduledEventEntityType.external`, this value is not ``None``.
"""
__slots__ = ("location",)
def __init__(self, *, location: Optional[str] = None):
self.location: Optional[str] = location
def __repr__(self) -> str:
return f"<GuildScheduledEventMetadata location={self.location!r}>"
def to_dict(self) -> GuildScheduledEventEntityMetadataPayload:
if self.location is not None:
return {"location": self.location}
return {}
@classmethod
def from_dict(
cls, data: GuildScheduledEventEntityMetadataPayload
) -> GuildScheduledEventMetadata:
return GuildScheduledEventMetadata(location=data.get("location"))
class GuildScheduledEvent(Hashable):
"""
Represents a guild scheduled event.
.. versionadded:: 2.3
.. container:: operations
.. describe:: x == y
Checks if two guild scheduled events are equal.
.. describe:: x != y
Checks if two guild scheduled events are not equal.
.. describe:: hash(x)
Returns the guild scheduled event's hash.
Attributes
----------
id: :class:`int`
The ID of the guild scheduled event.
guild_id: :class:`int`
The guild ID which the guild scheduled event belongs to.
channel_id: Optional[:class:`int`]
The channel ID in which the guild scheduled event will be hosted.
This field is ``None`` if :attr:`entity_type` is :class:`GuildScheduledEventEntityType.external`
creator_id: Optional[:class:`int`]
The ID of the user that created the guild scheduled event.
This field is ``None`` for events created before October 25th, 2021.
creator: Optional[:class:`User`]
The user that created the guild scheduled event.
This field is ``None`` for events created before October 25th, 2021.
name: :class:`str`
The name of the guild scheduled event (1-100 characters).
description: :class:`str`
The description of the guild scheduled event (1-1000 characters).
scheduled_start_time: :class:`datetime.datetime`
The time when the guild scheduled event will start
scheduled_end_time: Optional[:class:`datetime.datetime`]
The time when the guild scheduled event will end, or ``None`` if the event does not have a scheduled time to end.
privacy_level: :class:`GuildScheduledEventPrivacyLevel`
The privacy level of the guild scheduled event.
status: :class:`GuildScheduledEventStatus`
The status of the guild scheduled event.
entity_type: :class:`GuildScheduledEventEntityType`
The type of the guild scheduled event.
entity_id: Optional[:class:`int`]
The ID of an entity associated with the guild scheduled event.
entity_metadata: :class:`GuildScheduledEventMetadata`
Additional metadata for the guild scheduled event.
user_count: Optional[:class:`int`]
The number of users subscribed to the guild scheduled event.
If the guild scheduled event was fetched with ``with_user_count`` set to ``False``, this field is ``None``.
"""
__slots__ = (
"_state",
"id",
"guild_id",
"channel_id",
"creator_id",
"name",
"description",
"scheduled_start_time",
"scheduled_end_time",
"privacy_level",
"status",
"entity_type",
"entity_id",
"entity_metadata",
"creator",
"user_count",
"_image",
"_cs_guild",
"_cs_channel",
)
def __init__(self, *, state: ConnectionState, data: GuildScheduledEventPayload):
self._state: ConnectionState = state
self._update(data)
def _update(self, data: GuildScheduledEventPayload) -> None:
self.id: int = int(data["id"])
self.guild_id: int = int(data["guild_id"])
self.channel_id: Optional[int] = _get_as_snowflake(data, "channel_id")
self.creator_id: Optional[int] = _get_as_snowflake(data, "creator_id")
self.name: str = data["name"]
self.description: Optional[str] = data.get("description")
self.scheduled_start_time: datetime = parse_time(data["scheduled_start_time"])
self.scheduled_end_time: Optional[datetime] = parse_time(data["scheduled_end_time"])
self.privacy_level: GuildScheduledEventPrivacyLevel = try_enum(
GuildScheduledEventPrivacyLevel, data["privacy_level"]
)
self.status: GuildScheduledEventStatus = try_enum(GuildScheduledEventStatus, data["status"])
self.entity_type: GuildScheduledEventEntityType = try_enum(
GuildScheduledEventEntityType, data["entity_type"]
)
self.entity_id: Optional[int] = _get_as_snowflake(data, "entity_id")
metadata = data["entity_metadata"]
self.entity_metadata: Optional[GuildScheduledEventMetadata] = (
None if metadata is None else GuildScheduledEventMetadata.from_dict(metadata)
)
creator_data = data.get("creator")
self.creator: Optional[User]
if creator_data is not None:
self.creator = User(state=self._state, data=creator_data)
elif self.creator_id is not None:
self.creator = self._state.get_user(self.creator_id)
else:
self.creator = None
self.user_count: Optional[int] = data.get("user_count")
self._image: Optional[str] = data.get("image")
def __repr__(self) -> str:
return (
"<GuildScheduledEvent "
+ " ".join(
f"{attr}={getattr(self, attr)!r}"
for attr in self.__slots__
if not attr.startswith("_")
)
+ ">"
)
def __str__(self) -> str:
return self.name
@cached_slot_property("_cs_guild")
def guild(self) -> Optional[Guild]:
"""Optional[:class:`Guild`]: The guild which the guild scheduled event belongs to."""
return self._state._get_guild(self.guild_id)
@cached_slot_property("_cs_channel")
def channel(self) -> Optional[GuildChannel]:
"""Optional[:class:`abc.GuildChannel`]: The channel in which the guild scheduled event will be hosted."""
if self.channel_id is None:
return None
guild = self.guild
return None if guild is None else guild.get_channel(self.channel_id)
@property
def image(self) -> Optional[Asset]:
"""Optional[:class:`Asset`]: The cover image asset of the guild scheduled event, if available."""
if self._image is None:
return None
return Asset._from_guild_scheduled_event_image(self._state, self.id, self._image)
async def delete(self) -> None:
"""|coro|
Deletes the guild scheduled event.
Raises
------
Forbidden
You do not have proper permissions to delete the event.
NotFound
The event does not exist.
HTTPException
Deleting the event failed.
"""
await self._state.http.delete_guild_scheduled_event(self.guild_id, self.id)
async def edit(
self,
*,
name: str = MISSING,
description: Optional[str] = MISSING,
image: Optional[AssetBytes] = MISSING,
channel_id: Optional[int] = MISSING,
privacy_level: GuildScheduledEventPrivacyLevel = MISSING,
scheduled_start_time: datetime = MISSING,
scheduled_end_time: datetime = MISSING,
entity_type: GuildScheduledEventEntityType = MISSING,
entity_metadata: Optional[GuildScheduledEventMetadata] = MISSING,
status: GuildScheduledEventStatus = MISSING,
reason: Optional[str] = None,
) -> GuildScheduledEvent:
"""|coro|
Edits the guild scheduled event.
If updating ``entity_type`` to :class:`GuildScheduledEventEntityType.external`:
- ``channel_id`` should be set to ``None`` or ignored
- ``entity_metadata`` with a location field must be provided
- ``scheduled_end_time`` must be provided
Parameters
----------
name: :class:`str`
The name of the guild scheduled event.
description: Optional[:class:`str`]
The description of the guild scheduled event.
image: Optional[|resource_type|]
The cover image of the guild scheduled event. Set to ``None`` to remove the image.
.. versionadded:: 2.4
.. versionchanged:: 2.5
Now accepts various resource types in addition to :class:`bytes`.
channel_id: Optional[:class:`int`]
The channel ID in which the guild scheduled event will be hosted.
Set to ``None`` if changing ``entity_type`` to :class:`GuildScheduledEventEntityType.external`.
privacy_level: :class:`GuildScheduledEventPrivacyLevel`
The privacy level of the guild scheduled event.
scheduled_start_time: :class:`datetime.datetime`
The time to schedule the guild scheduled event.
scheduled_end_time: :class:`datetime.datetime`
The time when the guild scheduled event is scheduled to end.
entity_type: :class:`GuildScheduledEventEntityType`
The entity type of the guild scheduled event.
entity_metadata: Optional[:class:`GuildScheduledEventMetadata`]
The entity metadata of the guild scheduled event.
status: :class:`GuildScheduledEventStatus`
The status of the guild scheduled event.
reason: Optional[:class:`str`]
The reason for editing the guild scheduled event. Shows up on the audit log.
Raises
------
Forbidden
You do not have proper permissions to edit the event.
NotFound
The event does not exist or the ``image`` asset couldn't be found.
HTTPException
Editing the event failed.
TypeError
The ``image`` asset is a lottie sticker (see :func:`Sticker.read`).
Returns
-------
:class:`GuildScheduledEvent`
The newly updated guild scheduled event instance.
"""
fields: Dict[str, Any] = {}
is_external = entity_type is GuildScheduledEventEntityType.external
error_for_external_entity = (
"if entity_type is GuildScheduledEventEntityType.external, {} must be {}"
)
if privacy_level is not MISSING:
if not isinstance(privacy_level, GuildScheduledEventPrivacyLevel):
raise ValueError(
"privacy_level must be an instance of GuildScheduledEventPrivacyLevel"
)
fields["privacy_level"] = privacy_level.value
if entity_type is not MISSING:
if not isinstance(entity_type, GuildScheduledEventEntityType):
raise ValueError("entity_type must be an instance of GuildScheduledEventEntityType")
fields["entity_type"] = entity_type.value
if not entity_metadata and is_external:
raise ValueError(error_for_external_entity.format("entity_metadata", "provided"))
if entity_metadata is not MISSING:
if entity_metadata is None:
fields["entity_metadata"] = None
elif isinstance(entity_metadata, GuildScheduledEventMetadata):
fields["entity_metadata"] = entity_metadata.to_dict()
else:
raise ValueError(
"entity_metadata must be an instance of GuildScheduledEventMetadata"
)
if status is not MISSING:
if not isinstance(status, GuildScheduledEventStatus):
raise ValueError("status must be an instance of GuildScheduledEventStatus")
fields["status"] = status.value
if name is not MISSING:
fields["name"] = name
if description is not MISSING:
fields["description"] = description
if image is not MISSING:
fields["image"] = await _assetbytes_to_base64_data(image)
if channel_id is not MISSING:
if channel_id is not None and is_external:
raise ValueError(
error_for_external_entity.format("channel_id", "None or not provided")
)
fields["channel_id"] = channel_id
elif channel_id is None and is_external:
fields["channel_id"] = None
if scheduled_start_time is not MISSING:
fields["scheduled_start_time"] = scheduled_start_time.isoformat()
if scheduled_end_time is not MISSING:
fields["scheduled_end_time"] = scheduled_end_time.isoformat()
elif is_external:
raise ValueError(error_for_external_entity.format("scheduled_end_time", "provided"))
data = await self._state.http.edit_guild_scheduled_event(
guild_id=self.guild_id, event_id=self.id, reason=reason, **fields
)
return GuildScheduledEvent(state=self._state, data=data)
def fetch_users(
self,
*,
limit: Optional[int] = None,
with_members: bool = True,
before: Optional[Snowflake] = None,
after: Optional[Snowflake] = None,
) -> GuildScheduledEventUserIterator:
"""|coro|
Returns an :class:`AsyncIterator` of users subscribed to the guild scheduled event.
If ``before`` is specified, users are returned in reverse order,
i.e. starting with the highest ID.
.. versionchanged:: 2.5
Now returns an :class:`AsyncIterator` instead of a list of the first 100 users.
Parameters
----------
limit: Optional[:class:`int`]
The number of users to retrieve.
with_members: :class:`bool`
Whether to include some users as members. Defaults to ``True``.
before: Optional[:class:`abc.Snowflake`]
Retrieve users before this object.
after: Optional[:class:`abc.Snowflake`]
Retrieve users after this object.
Raises
------
Forbidden
You do not have proper permissions to fetch the users.
NotFound
The event does not exist.
HTTPException
Retrieving the users failed.
Yields
------
Union[:class:`User`, :class:`Member`]
The member (if retrievable) or user subscribed to the guild scheduled event.
Examples
--------
Usage ::
async for user in event.fetch_users(limit=500):
print(user.name)
Flattening into a list ::
users = await event.fetch_users(limit=250).flatten()
"""
return GuildScheduledEventUserIterator(
self,
limit=limit,
with_members=with_members,
before=before,
after=after,
)