committed by
GitHub
22 changed files with 1313 additions and 110 deletions
@ -0,0 +1,549 @@ |
|||||
|
""" |
||||
|
The MIT License (MIT) |
||||
|
|
||||
|
Copyright (c) 2015-present Rapptz |
||||
|
|
||||
|
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, AsyncIterator, Dict, Optional, Union |
||||
|
|
||||
|
from .asset import Asset |
||||
|
from .enums import EventStatus, EntityType, PrivacyLevel, try_enum |
||||
|
from .mixins import Hashable |
||||
|
from .object import Object, OLDEST_OBJECT |
||||
|
from .utils import parse_time, _get_as_snowflake, _bytes_to_base64_data, MISSING |
||||
|
|
||||
|
if TYPE_CHECKING: |
||||
|
from .types.scheduled_event import ( |
||||
|
GuildScheduledEvent as GuildScheduledEventPayload, |
||||
|
GuildScheduledEventWithUserCount as GuildScheduledEventWithUserCountPayload, |
||||
|
EntityMetadata, |
||||
|
) |
||||
|
|
||||
|
from .abc import Snowflake |
||||
|
from .channel import VoiceChannel, StageChannel |
||||
|
from .state import ConnectionState |
||||
|
from .user import User |
||||
|
|
||||
|
GuildScheduledEventPayload = Union[GuildScheduledEventPayload, GuildScheduledEventWithUserCountPayload] |
||||
|
|
||||
|
# fmt: off |
||||
|
__all__ = ( |
||||
|
"ScheduledEvent", |
||||
|
) |
||||
|
# fmt: on |
||||
|
|
||||
|
|
||||
|
class ScheduledEvent(Hashable): |
||||
|
"""Represents a scheduled event in a guild. |
||||
|
|
||||
|
.. versionadded:: 2.0 |
||||
|
|
||||
|
.. container:: operations |
||||
|
|
||||
|
.. describe:: x == y |
||||
|
|
||||
|
Checks if two scheduled events are equal. |
||||
|
|
||||
|
.. describe:: x != y |
||||
|
|
||||
|
Checks if two scheduled events are not equal. |
||||
|
|
||||
|
.. describe:: hash(x) |
||||
|
|
||||
|
Returns the scheduled event's hash. |
||||
|
|
||||
|
Attributes |
||||
|
---------- |
||||
|
id: :class:`int` |
||||
|
The scheduled event's ID. |
||||
|
name: :class:`str` |
||||
|
The name of the scheduled event. |
||||
|
description: :class:`str` |
||||
|
The description of the scheduled event. |
||||
|
entity_type: :class:`EntityType` |
||||
|
The type of entity this event is for. |
||||
|
entity_id: :class:`int` |
||||
|
The ID of the entity this event is for. |
||||
|
start_time: :class:`datetime.datetime` |
||||
|
The time that the scheduled event will start in UTC. |
||||
|
end_time: :class:`datetime.datetime` |
||||
|
The time that the scheduled event will end in UTC. |
||||
|
privacy_level: :class:`PrivacyLevel` |
||||
|
The privacy level of the scheduled event. |
||||
|
status: :class:`EventStatus` |
||||
|
The status of the scheduled event. |
||||
|
user_count: :class:`int` |
||||
|
The number of users subscribed to the scheduled event. |
||||
|
creator: Optional[:class:`User`] |
||||
|
The user that created the scheduled event. |
||||
|
location: Optional[:class:`str`] |
||||
|
The location of the scheduled event. |
||||
|
""" |
||||
|
|
||||
|
__slots__ = ( |
||||
|
'_state', |
||||
|
'_users', |
||||
|
'id', |
||||
|
'guild_id', |
||||
|
'name', |
||||
|
'description', |
||||
|
'entity_type', |
||||
|
'entity_id', |
||||
|
'start_time', |
||||
|
'end_time', |
||||
|
'privacy_level', |
||||
|
'status', |
||||
|
'_cover_image', |
||||
|
'user_count', |
||||
|
'creator', |
||||
|
'channel_id', |
||||
|
'location', |
||||
|
) |
||||
|
|
||||
|
def __init__(self, *, state: ConnectionState, data: GuildScheduledEventPayload) -> None: |
||||
|
self._state = state |
||||
|
self._users: Dict[int, User] = {} |
||||
|
self._update(data) |
||||
|
|
||||
|
def _update(self, data: GuildScheduledEventPayload) -> None: |
||||
|
self.id: int = int(data['id']) |
||||
|
self.guild_id: int = int(data['guild_id']) |
||||
|
self.name: str = data['name'] |
||||
|
self.description: str = data.get('description', '') |
||||
|
self.entity_type = try_enum(EntityType, data['entity_type']) |
||||
|
self.entity_id: int = int(data['id']) |
||||
|
self.start_time: datetime = parse_time(data['scheduled_start_time']) |
||||
|
self.privacy_level: PrivacyLevel = try_enum(PrivacyLevel, data['status']) |
||||
|
self.status: EventStatus = try_enum(EventStatus, data['status']) |
||||
|
self._cover_image: Optional[str] = data['image'] |
||||
|
self.user_count: int = data.get('user_count', 0) |
||||
|
|
||||
|
creator = data.get('creator') |
||||
|
self.creator: Optional[User] = self._state.store_user(creator) if creator else None |
||||
|
|
||||
|
self.end_time: Optional[datetime] = parse_time(data.get('scheduled_end_time')) |
||||
|
self.channel_id: Optional[int] = _get_as_snowflake(data, 'channel_id') |
||||
|
|
||||
|
metadata = data.get('metadata') |
||||
|
if metadata: |
||||
|
self._unroll_metadata(metadata) |
||||
|
|
||||
|
def _unroll_metadata(self, data: EntityMetadata): |
||||
|
self.location: Optional[str] = data.get('location') |
||||
|
|
||||
|
@classmethod |
||||
|
def from_creation(cls, *, state: ConnectionState, data: GuildScheduledEventPayload): |
||||
|
creator_id = data.get('creator_id') |
||||
|
self = cls(state=state, data=data) |
||||
|
if creator_id: |
||||
|
self.creator = self._state.get_user(int(creator_id)) |
||||
|
|
||||
|
def __repr__(self) -> str: |
||||
|
return f'<GuildScheduledEvent id={self.id} name={self.name!r} guild_id={self.guild_id!r} creator={self.creator!r}>' |
||||
|
|
||||
|
@property |
||||
|
def cover_image(self) -> Optional[Asset]: |
||||
|
"""Optional[:class:`Asset`]: The scheduled event's cover image.""" |
||||
|
if self._cover_image is None: |
||||
|
return None |
||||
|
return Asset._from_scheduled_event_cover_image(self._state, self.id, self._cover_image) |
||||
|
|
||||
|
@property |
||||
|
def channel(self) -> Optional[Union[VoiceChannel, StageChannel]]: |
||||
|
"""Optional[Union[:class:`VoiceChannel`, :class:`StageChannel`]]: The channel this scheduled event is in.""" |
||||
|
return self.guild.get_channel(self.channel_id) # type: ignore |
||||
|
|
||||
|
async def start(self, *, reason: Optional[str] = None) -> ScheduledEvent: |
||||
|
"""|coro| |
||||
|
|
||||
|
Starts the scheduled event. |
||||
|
|
||||
|
Shorthand for: |
||||
|
|
||||
|
.. code-block:: python3 |
||||
|
|
||||
|
await event.edit(status=EventStatus.active) |
||||
|
|
||||
|
Parameters |
||||
|
----------- |
||||
|
reason: Optional[:class:`str`] |
||||
|
The reason for starting the scheduled event. |
||||
|
|
||||
|
Raises |
||||
|
------ |
||||
|
ValueError |
||||
|
The scheduled event has already started or has ended. |
||||
|
Forbidden |
||||
|
You do not have the proper permissions to start the scheduled event. |
||||
|
HTTPException |
||||
|
The scheduled event could not be started. |
||||
|
|
||||
|
Returns |
||||
|
------- |
||||
|
:class:`ScheduledEvent` |
||||
|
The scheduled event that was started. |
||||
|
""" |
||||
|
if self.status is not EventStatus.scheduled: |
||||
|
raise ValueError('This scheduled event is already running.') |
||||
|
|
||||
|
return await self.edit(status=EventStatus.active, reason=reason) |
||||
|
|
||||
|
async def end(self, *, reason: Optional[str] = None) -> ScheduledEvent: |
||||
|
"""|coro| |
||||
|
|
||||
|
Ends the scheduled event. |
||||
|
|
||||
|
Shorthand for: |
||||
|
|
||||
|
.. code-block:: python3 |
||||
|
|
||||
|
await event.edit(status=EventStatus.completed) |
||||
|
|
||||
|
Parameters |
||||
|
----------- |
||||
|
reason: Optional[:class:`str`] |
||||
|
The reason for ending the scheduled event. |
||||
|
|
||||
|
Raises |
||||
|
------ |
||||
|
ValueError |
||||
|
The scheduled event is not active or has already ended. |
||||
|
Forbidden |
||||
|
You do not have the proper permissions to end the scheduled event. |
||||
|
HTTPException |
||||
|
The scheduled event could not be ended. |
||||
|
|
||||
|
Returns |
||||
|
------- |
||||
|
:class:`ScheduledEvent` |
||||
|
The scheduled event that was ended. |
||||
|
""" |
||||
|
if self.status is not EventStatus.active: |
||||
|
raise ValueError('This scheduled event is not active.') |
||||
|
|
||||
|
return await self.edit(status=EventStatus.ended, reason=reason) |
||||
|
|
||||
|
async def cancel(self, *, reason: Optional[str] = None) -> ScheduledEvent: |
||||
|
"""|coro| |
||||
|
|
||||
|
Cancels the scheduled event. |
||||
|
|
||||
|
Shorthand for: |
||||
|
|
||||
|
.. code-block:: python3 |
||||
|
|
||||
|
await event.edit(status=EventStatus.cancelled) |
||||
|
|
||||
|
Parameters |
||||
|
----------- |
||||
|
reason: Optional[:class:`str`] |
||||
|
The reason for cancelling the scheduled event. |
||||
|
|
||||
|
Raises |
||||
|
------ |
||||
|
ValueError |
||||
|
The scheduled event is already running. |
||||
|
Forbidden |
||||
|
You do not have the proper permissions to cancel the scheduled event. |
||||
|
HTTPException |
||||
|
The scheduled event could not be cancelled. |
||||
|
|
||||
|
Returns |
||||
|
------- |
||||
|
:class:`ScheduledEvent` |
||||
|
The scheduled event that was cancelled. |
||||
|
""" |
||||
|
if self.status is not EventStatus.scheduled: |
||||
|
raise ValueError('This scheduled event is already running.') |
||||
|
|
||||
|
return await self.edit(status=EventStatus.cancelled, reason=reason) |
||||
|
|
||||
|
async def edit( |
||||
|
self, |
||||
|
*, |
||||
|
name: str = MISSING, |
||||
|
description: str = MISSING, |
||||
|
channel: Optional[Snowflake] = MISSING, |
||||
|
start_time: datetime = MISSING, |
||||
|
end_time: datetime = MISSING, |
||||
|
privacy_level: PrivacyLevel = MISSING, |
||||
|
entity_type: EntityType = MISSING, |
||||
|
status: EventStatus = MISSING, |
||||
|
image: bytes = MISSING, |
||||
|
location: str = MISSING, |
||||
|
reason: Optional[str] = None, |
||||
|
) -> ScheduledEvent: |
||||
|
r"""|coro| |
||||
|
|
||||
|
Edits the scheduled event. |
||||
|
|
||||
|
Requires :attr:`~Permissions.manage_events` permissions. |
||||
|
|
||||
|
Parameters |
||||
|
----------- |
||||
|
name: :class:`str` |
||||
|
The name of the scheduled event. |
||||
|
description: :class:`str` |
||||
|
The description of the scheduled event. |
||||
|
channel: Optional[:class:`~discord.abc.Snowflake`] |
||||
|
The channel to put the scheduled event in. |
||||
|
|
||||
|
Required if the entity type is either :attr:`EntityType.voice` or |
||||
|
:attr:`EntityType.stage_instance`. |
||||
|
start_time: :class:`datetime.datetime` |
||||
|
The time that the scheduled event will start. This must be a timezone-aware |
||||
|
datetime object. Consider using :func:`utils.utcnow`. |
||||
|
end_time: :class:`datetime.datetime` |
||||
|
The time that the scheduled event will end. This must be a timezone-aware |
||||
|
datetime object. Consider using :func:`utils.utcnow`. |
||||
|
|
||||
|
Required if the entity type is :attr:`EntityType.external`. |
||||
|
privacy_level: :class:`PrivacyLevel` |
||||
|
The privacy level of the scheduled event. |
||||
|
entity_type: :class:`EntityType` |
||||
|
The new entity type. |
||||
|
status: :class:`EventStatus` |
||||
|
The new status of the scheduled event. |
||||
|
image: :class:`bytes` |
||||
|
The new image of the scheduled event. |
||||
|
location: :class:`str` |
||||
|
The new location of the scheduled event. |
||||
|
|
||||
|
Required if the entity type is :attr:`EntityType.external`. |
||||
|
reason: Optional[:class:`str`] |
||||
|
The reason for editing the scheduled event. Shows up on the audit log. |
||||
|
|
||||
|
Raises |
||||
|
------- |
||||
|
TypeError |
||||
|
`image` was not a :term:`py:bytes-like object`, or ``privacy_level`` |
||||
|
was not a :class:`PrivacyLevel`, or ``entity_type`` was not an |
||||
|
:class:`EntityType`, ``status`` was not an :class:`EventStatus`, or |
||||
|
an argument was provided that was incompatible with the scheduled event's |
||||
|
entity type. |
||||
|
ValueError |
||||
|
``start_time`` or ``end_time`` was not a timezone-aware datetime object. |
||||
|
Forbidden |
||||
|
You do not have permissions to edit the scheduled event. |
||||
|
HTTPException |
||||
|
Editing the scheduled event failed. |
||||
|
|
||||
|
Returns |
||||
|
-------- |
||||
|
:class:`ScheduledEvent` |
||||
|
The edited scheduled event. |
||||
|
""" |
||||
|
payload = {} |
||||
|
metadata = {} |
||||
|
|
||||
|
if name is not MISSING: |
||||
|
payload['name'] = name |
||||
|
|
||||
|
if start_time is not MISSING: |
||||
|
if start_time.tzinfo is None: |
||||
|
raise ValueError( |
||||
|
'start_time must be an aware datetime. Consider using discord.utils.utcnow() or datetime.datetime.now().astimezone() for local time.' |
||||
|
) |
||||
|
payload['scheduled_start_time'] = start_time.isoformat() |
||||
|
|
||||
|
if description is not MISSING: |
||||
|
payload['description'] = description |
||||
|
|
||||
|
if privacy_level is not MISSING: |
||||
|
if not isinstance(privacy_level, PrivacyLevel): |
||||
|
raise TypeError('privacy_level must be of type PrivacyLevel.') |
||||
|
|
||||
|
payload['privacy_level'] = privacy_level.value |
||||
|
|
||||
|
if status is not MISSING: |
||||
|
if not isinstance(status, EventStatus): |
||||
|
raise TypeError('status must be of type EventStatus') |
||||
|
|
||||
|
payload['status'] = status.value |
||||
|
|
||||
|
if image is not MISSING: |
||||
|
image_as_str: str = _bytes_to_base64_data(image) |
||||
|
payload['image'] = image_as_str |
||||
|
|
||||
|
if entity_type is not MISSING: |
||||
|
if not isinstance(entity_type, EntityType): |
||||
|
raise TypeError('entity_type must be of type EntityType') |
||||
|
|
||||
|
payload['entity_type'] = entity_type.value |
||||
|
|
||||
|
_entity_type = entity_type or self.entity_type |
||||
|
|
||||
|
if _entity_type in (EntityType.stage_instance, EntityType.voice): |
||||
|
if channel is MISSING or channel is None: |
||||
|
raise TypeError('channel must be set when entity_type is voice or stage_instance') |
||||
|
|
||||
|
payload['channel_id'] = channel.id |
||||
|
|
||||
|
if location is not MISSING: |
||||
|
raise TypeError('location cannot be set when entity_type is voice or stage_instance') |
||||
|
else: |
||||
|
if channel is not MISSING: |
||||
|
raise TypeError('channel cannot be set when entity_type is external') |
||||
|
|
||||
|
if location is MISSING or location is None: |
||||
|
raise TypeError('location must be set when entity_type is external') |
||||
|
|
||||
|
metadata['location'] = location |
||||
|
|
||||
|
if end_time is MISSING: |
||||
|
raise TypeError('end_time must be set when entity_type is external') |
||||
|
|
||||
|
if end_time.tzinfo is None: |
||||
|
raise ValueError( |
||||
|
'end_time must be an aware datetime. Consider using discord.utils.utcnow() or datetime.datetime.now().astimezone() for local time.' |
||||
|
) |
||||
|
payload['scheduled_end_time'] = end_time.isoformat() |
||||
|
|
||||
|
if metadata: |
||||
|
payload['entity_metadata'] = metadata |
||||
|
|
||||
|
data = await self._state.http.edit_scheduled_event(self.guild_id, self.id, **payload, reason=reason) |
||||
|
s = ScheduledEvent(state=self._state, data=data) |
||||
|
s._users = self._users |
||||
|
return s |
||||
|
|
||||
|
async def delete(self, *, reason: Optional[str] = None) -> None: |
||||
|
"""|coro| |
||||
|
|
||||
|
Deletes the scheduled event. |
||||
|
|
||||
|
Requires :attr:`~Permissions.manage_events` permissions. |
||||
|
|
||||
|
Parameters |
||||
|
----------- |
||||
|
reason: Optional[:class:`str`] |
||||
|
The reason for deleting the scheduled event. Shows up on the audit log. |
||||
|
|
||||
|
Raises |
||||
|
------ |
||||
|
Forbidden |
||||
|
You do not have permissions to delete the scheduled event. |
||||
|
HTTPException |
||||
|
Deleting the scheduled event failed. |
||||
|
""" |
||||
|
await self._state.http.delete_scheduled_event(self.guild_id, self.id, reason=reason) |
||||
|
|
||||
|
async def users( |
||||
|
self, |
||||
|
*, |
||||
|
limit: Optional[int] = None, |
||||
|
before: Optional[Snowflake] = None, |
||||
|
after: Optional[Snowflake] = None, |
||||
|
oldest_first: bool = MISSING, |
||||
|
) -> AsyncIterator[User]: |
||||
|
"""|coro| |
||||
|
|
||||
|
Retrieves all :class:`User` that are in this thread. |
||||
|
|
||||
|
This requires :attr:`Intents.members` to get information about members |
||||
|
other than yourself. |
||||
|
|
||||
|
Raises |
||||
|
------- |
||||
|
HTTPException |
||||
|
Retrieving the members failed. |
||||
|
|
||||
|
Returns |
||||
|
-------- |
||||
|
List[:class:`User`] |
||||
|
All thread members in the thread. |
||||
|
""" |
||||
|
|
||||
|
async def _before_strategy(retrieve, before, limit): |
||||
|
before_id = before.id if before else None |
||||
|
users = await self._state.http.get_scheduled_event_users( |
||||
|
self.guild_id, self.id, limit=retrieve, with_member=False, before=before_id |
||||
|
) |
||||
|
|
||||
|
if users: |
||||
|
if limit is not None: |
||||
|
limit -= len(users) |
||||
|
|
||||
|
before = Object(id=users[-1]['user']['id']) |
||||
|
|
||||
|
return users, before, limit |
||||
|
|
||||
|
async def _after_strategy(retrieve, after, limit): |
||||
|
after_id = after.id if after else None |
||||
|
users = await self._state.http.get_scheduled_event_users( |
||||
|
self.guild_id, self.id, limit=retrieve, with_member=False, after=after_id |
||||
|
) |
||||
|
|
||||
|
if users: |
||||
|
if limit is not None: |
||||
|
limit -= len(users) |
||||
|
|
||||
|
after = Object(id=users[0]['user']['id']) |
||||
|
|
||||
|
return users, after, limit |
||||
|
|
||||
|
if limit is None: |
||||
|
limit = self.user_count or None |
||||
|
|
||||
|
if oldest_first is MISSING: |
||||
|
reverse = after is not None |
||||
|
else: |
||||
|
reverse = oldest_first |
||||
|
|
||||
|
predicate = None |
||||
|
|
||||
|
if reverse: |
||||
|
strategy, state = _after_strategy, after |
||||
|
if before: |
||||
|
predicate = lambda u: u['user']['id'] < before.id |
||||
|
else: |
||||
|
strategy, state = _before_strategy, before |
||||
|
if after and after != OLDEST_OBJECT: |
||||
|
predicate = lambda u: u['user']['id'] > after.id |
||||
|
|
||||
|
while True: |
||||
|
retrieve = min(100 if limit is None else limit, 100) |
||||
|
if retrieve < 1: |
||||
|
return |
||||
|
|
||||
|
data, state, limit = await strategy(retrieve, state, limit) |
||||
|
|
||||
|
if len(data) < 100: |
||||
|
limit = 0 |
||||
|
|
||||
|
if reverse: |
||||
|
data = reversed(data) |
||||
|
if predicate: |
||||
|
data = filter(predicate, data) |
||||
|
|
||||
|
users = (self._state.store_user(raw_user['user']) for raw_user in data) |
||||
|
|
||||
|
for user in users: |
||||
|
yield user |
||||
|
|
||||
|
def _add_user(self, user: User) -> None: |
||||
|
self._users[user.id] = user |
||||
|
|
||||
|
def _pop_user(self, user_id: int) -> None: |
||||
|
self._users.pop(user_id) |
Loading…
Reference in new issue