Source code for discord.template

"""
The MIT License (MIT)

Copyright (c) 2015-2021 Rapptz
Copyright (c) 2021-present Pycord 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 typing import TYPE_CHECKING, Any

from .guild import Guild
from .utils import MISSING, _bytes_to_base64_data, parse_time

__all__ = ("Template",)

if TYPE_CHECKING:
    import datetime

    from .state import ConnectionState
    from .types.template import Template as TemplatePayload
    from .user import User


class _FriendlyHttpAttributeErrorHelper:
    __slots__ = ()

    def __getattr__(self, attr):
        raise AttributeError("PartialTemplateState does not support http methods.")


class _PartialTemplateState:
    def __init__(self, *, state):
        self.__state = state
        self.http = _FriendlyHttpAttributeErrorHelper()

    @property
    def shard_count(self):
        return self.__state.shard_count

    @property
    def user(self):
        return self.__state.user

    @property
    def self_id(self):
        return self.__state.user.id

    @property
    def member_cache_flags(self):
        return self.__state.member_cache_flags

    def store_emoji(self, guild, packet):
        return None

    def _get_voice_client(self, id):
        return None

    def _get_message(self, id):
        return None

    def _get_guild(self, id):
        return self.__state._get_guild(id)

    async def query_members(self, **kwargs: Any):
        return []

    def __getattr__(self, attr):
        raise AttributeError(f"PartialTemplateState does not support {attr!r}.")


[docs]class Template: """Represents a Discord template. .. versionadded:: 1.4 Attributes ---------- code: :class:`str` The template code. uses: :class:`int` How many times the template has been used. name: :class:`str` The name of the template. description: :class:`str` The description of the template. creator: :class:`User` The creator of the template. created_at: :class:`datetime.datetime` An aware datetime in UTC representing when the template was created. updated_at: :class:`datetime.datetime` An aware datetime in UTC representing when the template was last updated. This is referred to as "last synced" in the official Discord client. source_guild: :class:`Guild` The source guild. is_dirty: Optional[:class:`bool`] Whether the template has unsynced changes. .. versionadded:: 2.0 """ __slots__ = ( "code", "uses", "name", "description", "creator", "created_at", "updated_at", "source_guild", "is_dirty", "_state", ) def __init__(self, *, state: ConnectionState, data: TemplatePayload) -> None: self._state = state self._store(data) def _store(self, data: TemplatePayload) -> None: self.code: str = data["code"] self.uses: int = data["usage_count"] self.name: str = data["name"] self.description: str | None = data["description"] creator_data = data.get("creator") self.creator: User | None = ( None if creator_data is None else self._state.create_user(creator_data) ) self.created_at: datetime.datetime | None = parse_time(data.get("created_at")) self.updated_at: datetime.datetime | None = parse_time(data.get("updated_at")) guild_id = int(data["source_guild_id"]) guild: Guild | None = self._state._get_guild(guild_id) self.source_guild: Guild if guild is None: source_serialised = data["serialized_source_guild"] source_serialised["id"] = guild_id state = _PartialTemplateState(state=self._state) # Guild expects a ConnectionState, we're passing a _PartialTemplateState self.source_guild = Guild(data=source_serialised, state=state) # type: ignore else: self.source_guild = guild self.is_dirty: bool | None = data.get("is_dirty", None) def __repr__(self) -> str: return ( f"<Template code={self.code!r} uses={self.uses} name={self.name!r}" f" creator={self.creator!r} source_guild={self.source_guild!r} is_dirty={self.is_dirty}>" )
[docs] async def create_guild(self, name: str, icon: Any = None) -> Guild: """|coro| Creates a :class:`.Guild` using the template. Bot accounts in more than 10 guilds are not allowed to create guilds. Parameters ---------- name: :class:`str` The name of the guild. icon: :class:`bytes` The :term:`py:bytes-like object` representing the icon. See :meth:`.ClientUser.edit` for more details on what is expected. Returns ------- :class:`.Guild` The guild created. This is not the same guild that is added to cache. Raises ------ HTTPException Guild creation failed. InvalidArgument Invalid icon image format given. Must be PNG or JPG. """ if icon is not None: icon = _bytes_to_base64_data(icon) data = await self._state.http.create_from_template(self.code, name, icon) return Guild(data=data, state=self._state)
[docs] async def sync(self) -> Template: """|coro| Sync the template to the guild's current state. You must have the :attr:`~Permissions.manage_guild` permission in the source guild to do this. .. versionadded:: 1.7 .. versionchanged:: 2.0 The template is no longer synced in-place, instead it is returned. Returns ------- :class:`Template` The newly synced template. Raises ------ HTTPException Syncing the template failed. Forbidden You don't have permissions to sync the template. NotFound This template does not exist. """ data = await self._state.http.sync_template(self.source_guild.id, self.code) return Template(state=self._state, data=data)
[docs] async def edit( self, *, name: str = MISSING, description: str | None = MISSING, ) -> Template: """|coro| Edit the template metadata. You must have the :attr:`~Permissions.manage_guild` permission in the source guild to do this. .. versionadded:: 1.7 .. versionchanged:: 2.0 The template is no longer edited in-place, instead it is returned. Parameters ---------- name: :class:`str` The template's new name. description: Optional[:class:`str`] The template's new description. Returns ------- :class:`Template` The newly edited template. Raises ------ HTTPException Editing the template failed. Forbidden You don't have permissions to edit the template. NotFound This template does not exist. """ payload = {} if name is not MISSING: payload["name"] = name if description is not MISSING: payload["description"] = description data = await self._state.http.edit_template( self.source_guild.id, self.code, payload ) return Template(state=self._state, data=data)
[docs] async def delete(self) -> None: """|coro| Delete the template. You must have the :attr:`~Permissions.manage_guild` permission in the source guild to do this. .. versionadded:: 1.7 Raises ------ HTTPException Deleting the template failed. Forbidden You don't have permissions to delete the template. NotFound This template does not exist. """ await self._state.http.delete_template(self.source_guild.id, self.code)
@property def url(self) -> str: """The template url. .. versionadded:: 2.0 """ return f"https://discord.new/{self.code}"