mirror of
				https://github.com/Rapptz/discord.py.git
				synced 2025-10-25 02:23:04 +00:00 
			
		
		
		
	Implement Application Command Permissions models
This commit is contained in:
		| @@ -27,16 +27,20 @@ from datetime import datetime | ||||
|  | ||||
| from .errors import MissingApplicationID | ||||
| from ..permissions import Permissions | ||||
| from ..enums import AppCommandOptionType, AppCommandType, ChannelType, try_enum | ||||
| from ..enums import AppCommandOptionType, AppCommandType, AppCommandPermissionType, ChannelType, try_enum | ||||
| from ..mixins import Hashable | ||||
| from ..utils import _get_as_snowflake, parse_time, snowflake_time, MISSING | ||||
| from typing import Generic, List, TYPE_CHECKING, Optional, TypeVar, Union | ||||
| from ..object import Object | ||||
|  | ||||
| from typing import Any, Dict, Generic, List, TYPE_CHECKING, Optional, TypeVar, Union | ||||
|  | ||||
| __all__ = ( | ||||
|     'AppCommand', | ||||
|     'AppCommandGroup', | ||||
|     'AppCommandChannel', | ||||
|     'AppCommandThread', | ||||
|     'AppCommandPermissions', | ||||
|     'GuildAppCommandPermissions', | ||||
|     'Argument', | ||||
|     'Choice', | ||||
|     'AllChannels', | ||||
| @@ -54,6 +58,8 @@ if TYPE_CHECKING: | ||||
|         ApplicationCommand as ApplicationCommandPayload, | ||||
|         ApplicationCommandOptionChoice, | ||||
|         ApplicationCommandOption, | ||||
|         ApplicationCommandPermissions, | ||||
|         GuildApplicationCommandPermissions, | ||||
|     ) | ||||
|     from ..types.interactions import ( | ||||
|         PartialChannel, | ||||
| @@ -63,10 +69,15 @@ if TYPE_CHECKING: | ||||
|         ThreadMetadata, | ||||
|         ThreadArchiveDuration, | ||||
|     ) | ||||
|  | ||||
|     from ..abc import Snowflake | ||||
|     from ..state import ConnectionState | ||||
|     from ..guild import GuildChannel, Guild | ||||
|     from ..channel import TextChannel | ||||
|     from ..threads import Thread | ||||
|     from ..role import Role | ||||
|     from ..user import User | ||||
|     from ..member import Member | ||||
|  | ||||
|     ApplicationCommandParent = Union['AppCommand', 'AppCommandGroup'] | ||||
|  | ||||
| @@ -328,6 +339,45 @@ class AppCommand(Hashable): | ||||
|             ) | ||||
|         return AppCommand(data=data, state=state) | ||||
|  | ||||
|     async def fetch_permissions(self, guild: Snowflake) -> GuildAppCommandPermissions: | ||||
|         """|coro| | ||||
|  | ||||
|         Retrieves this command's permission in the guild. | ||||
|  | ||||
|         Parameters | ||||
|         ----------- | ||||
|         guild: :class:`~discord.abc.Snowflake` | ||||
|             The guild to retrieve the permissions from. | ||||
|  | ||||
|         Raises | ||||
|         ------- | ||||
|         Forbidden | ||||
|             You do not have permission to fetch the application command's permissions. | ||||
|         HTTPException | ||||
|             Fetching the application command's permissions failed. | ||||
|         MissingApplicationID | ||||
|             The client does not have an application ID. | ||||
|         NotFound | ||||
|             The application command's permissions could not be found. | ||||
|             This can also indicate that the permissions are synced with the guild | ||||
|             (i.e. they are unchanged from the default). | ||||
|  | ||||
|         Returns | ||||
|         -------- | ||||
|         :class:`GuildAppCommandPermissions` | ||||
|             An object representing the application command's permissions in the guild. | ||||
|         """ | ||||
|         state = self._state | ||||
|         if not state.application_id: | ||||
|             raise MissingApplicationID | ||||
|  | ||||
|         data = await state.http.get_application_command_permissions( | ||||
|             state.application_id, | ||||
|             guild.id, | ||||
|             self.id, | ||||
|         ) | ||||
|         return GuildAppCommandPermissions(data=data, state=state, command=self) | ||||
|  | ||||
|  | ||||
| class Choice(Generic[ChoiceT]): | ||||
|     """Represents an application command argument choice. | ||||
| @@ -804,6 +854,108 @@ class AppCommandGroup: | ||||
|         }  # type: ignore # Type checker does not understand this literal. | ||||
|  | ||||
|  | ||||
| class AppCommandPermissions: | ||||
|     """Represents the permissions for an application command. | ||||
|  | ||||
|     .. versionadded:: 2.0 | ||||
|  | ||||
|     Attributes | ||||
|     ----------- | ||||
|     guild: :class:`~discord.Guild` | ||||
|         The guild assosiated with this permission. | ||||
|     id: :class:`int` | ||||
|         The ID of the permission target, such as a role, channel, or guild. | ||||
|         The special ``guild_id - 1`` sentinel is used to represent "all channels". | ||||
|     target: Any | ||||
|         The role, user, or channel associated with this permission. This could also be the :class:`AllChannels` sentinel type. | ||||
|         Falls back to :class:`~discord.Object` if the target could not be found in the cache. | ||||
|     type: :class:`.AppCommandPermissionType` | ||||
|         The type of permission. | ||||
|     permission: :class:`bool` | ||||
|         The permission value. True for allow, False for deny. | ||||
|     """ | ||||
|  | ||||
|     __slots__ = ('id', 'type', 'permission', 'target', 'guild', '_state') | ||||
|  | ||||
|     def __init__(self, *, data: ApplicationCommandPermissions, guild: Optional[Guild], state: ConnectionState) -> None: | ||||
|         self._state: ConnectionState = state | ||||
|         self.guild: Optional[Guild] = guild | ||||
|  | ||||
|         self.id: int = int(data['id']) | ||||
|         self.type: AppCommandPermissionType = try_enum(AppCommandPermissionType, data['type']) | ||||
|         self.permission: bool = data['permission'] | ||||
|  | ||||
|         _object = None | ||||
|  | ||||
|         if self.type is AppCommandPermissionType.user: | ||||
|             if guild: | ||||
|                 _object = guild.get_member(self.id) | ||||
|             else: | ||||
|                 _object = self._state.get_user(self.id) | ||||
|         elif guild and self.type is AppCommandPermissionType.channel: | ||||
|             if self.id == (guild.id - 1): | ||||
|                 _object = AllChannels(guild) | ||||
|             else: | ||||
|                 _object = guild.get_channel(self.id) | ||||
|         elif guild and self.type is AppCommandPermissionType.role: | ||||
|             _object = guild.get_role(self.id) | ||||
|  | ||||
|         if _object is None: | ||||
|             _object = Object(id=self.id) | ||||
|  | ||||
|         self.target: Union[Object, User, Member, Role, AllChannels, GuildChannel] = _object | ||||
|  | ||||
|     def to_dict(self) -> ApplicationCommandPermissions: | ||||
|         return { | ||||
|             'id': self.target.id, | ||||
|             'type': self.type.value, | ||||
|             'permission': self.permission, | ||||
|         } | ||||
|  | ||||
|  | ||||
| class GuildAppCommandPermissions: | ||||
|     """Represents the permissions for an application command in a guild. | ||||
|  | ||||
|     .. versionadded:: 2.0 | ||||
|  | ||||
|     Attributes | ||||
|     ----------- | ||||
|     application_id: :class:`int` | ||||
|         The application ID. | ||||
|     command: :class:`.AppCommand` | ||||
|         The application command associated with the permissions. | ||||
|     id: :class:`int` | ||||
|         ID of the command or the application ID. | ||||
|         When this is the application ID instead of a command ID, | ||||
|         the permissions apply to all commands that do not contain explicit overwrites. | ||||
|     guild_id: :class:`int` | ||||
|         The guild ID associated with the permissions. | ||||
|     permissions: List[:class:`AppCommandPermissions`] | ||||
|        The permissions, this is a max of 100. | ||||
|     """ | ||||
|  | ||||
|     __slots__ = ('id', 'application_id', 'command', 'guild_id', 'permissions', '_state') | ||||
|  | ||||
|     def __init__(self, *, data: GuildApplicationCommandPermissions, state: ConnectionState, command: AppCommand) -> None: | ||||
|         self._state: ConnectionState = state | ||||
|         self.command: AppCommand = command | ||||
|  | ||||
|         self.id: int = int(data['id']) | ||||
|         self.application_id: int = int(data['application_id']) | ||||
|         self.guild_id: int = int(data['guild_id']) | ||||
|         self.permissions: List[AppCommandPermissions] = [ | ||||
|             AppCommandPermissions(data=value, guild=self.guild, state=self._state) for value in data['permissions'] | ||||
|         ] | ||||
|  | ||||
|     def to_dict(self) -> Dict[str, Any]: | ||||
|         return {'permissions': [p.to_dict() for p in self.permissions]} | ||||
|  | ||||
|     @property | ||||
|     def guild(self) -> Optional[Guild]: | ||||
|         """Optional[:class:`~discord.Guild`]: The guild associated with the permissions.""" | ||||
|         return self._state._get_guild(self.guild_id) | ||||
|  | ||||
|  | ||||
| def app_command_option_factory( | ||||
|     parent: ApplicationCommandParent, data: ApplicationCommandOption, *, state: Optional[ConnectionState] = None | ||||
| ) -> Union[Argument, AppCommandGroup]: | ||||
|   | ||||
		Reference in New Issue
	
	Block a user