mirror of
				https://github.com/Rapptz/discord.py.git
				synced 2025-10-25 18:43:00 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			287 lines
		
	
	
		
			8.2 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			287 lines
		
	
	
		
			8.2 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| """
 | |
| 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
 | |
| 
 | |
| import inspect
 | |
| from operator import attrgetter
 | |
| from typing import TYPE_CHECKING, Any, Literal, Optional, OrderedDict, Union, Protocol
 | |
| 
 | |
| from discord.utils import MISSING, maybe_coroutine
 | |
| 
 | |
| from .errors import NoPrivateMessage
 | |
| from .converter import GuildConverter
 | |
| 
 | |
| from discord import (
 | |
|     Member,
 | |
|     User,
 | |
|     TextChannel,
 | |
|     VoiceChannel,
 | |
|     DMChannel,
 | |
|     Thread,
 | |
| )
 | |
| 
 | |
| if TYPE_CHECKING:
 | |
|     from typing_extensions import Self
 | |
| 
 | |
|     from discord import Guild
 | |
| 
 | |
|     from .context import Context
 | |
| 
 | |
| __all__ = (
 | |
|     'Parameter',
 | |
|     'parameter',
 | |
|     'param',
 | |
|     'Author',
 | |
|     'CurrentChannel',
 | |
|     'CurrentGuild',
 | |
| )
 | |
| 
 | |
| 
 | |
| ParamKinds = Union[
 | |
|     Literal[inspect.Parameter.POSITIONAL_ONLY],
 | |
|     Literal[inspect.Parameter.POSITIONAL_OR_KEYWORD],
 | |
|     Literal[inspect.Parameter.VAR_POSITIONAL],
 | |
|     Literal[inspect.Parameter.KEYWORD_ONLY],
 | |
|     Literal[inspect.Parameter.VAR_KEYWORD],
 | |
| ]
 | |
| 
 | |
| empty: Any = inspect.Parameter.empty
 | |
| 
 | |
| 
 | |
| def _gen_property(name: str) -> property:
 | |
|     attr = f'_{name}'
 | |
|     return property(
 | |
|         attrgetter(attr),
 | |
|         lambda self, value: setattr(self, attr, value),
 | |
|         doc=f"The parameter's {name}.",
 | |
|     )
 | |
| 
 | |
| 
 | |
| class Parameter(inspect.Parameter):
 | |
|     r"""A class that stores information on a :class:`Command`\'s parameter.
 | |
| 
 | |
|     This is a subclass of :class:`inspect.Parameter`.
 | |
| 
 | |
|     .. versionadded:: 2.0
 | |
|     """
 | |
| 
 | |
|     __slots__ = ('_displayed_default', '_description', '_fallback')
 | |
| 
 | |
|     def __init__(
 | |
|         self,
 | |
|         name: str,
 | |
|         kind: ParamKinds,
 | |
|         default: Any = empty,
 | |
|         annotation: Any = empty,
 | |
|         description: str = empty,
 | |
|         displayed_default: str = empty,
 | |
|     ) -> None:
 | |
|         super().__init__(name=name, kind=kind, default=default, annotation=annotation)
 | |
|         self._name = name
 | |
|         self._kind = kind
 | |
|         self._description = description
 | |
|         self._default = default
 | |
|         self._annotation = annotation
 | |
|         self._displayed_default = displayed_default
 | |
|         self._fallback = False
 | |
| 
 | |
|     def replace(
 | |
|         self,
 | |
|         *,
 | |
|         name: str = MISSING,  # MISSING here cause empty is valid
 | |
|         kind: ParamKinds = MISSING,
 | |
|         default: Any = MISSING,
 | |
|         annotation: Any = MISSING,
 | |
|         description: str = MISSING,
 | |
|         displayed_default: Any = MISSING,
 | |
|     ) -> Self:
 | |
|         if name is MISSING:
 | |
|             name = self._name
 | |
|         if kind is MISSING:
 | |
|             kind = self._kind  # type: ignore  # this assignment is actually safe
 | |
|         if default is MISSING:
 | |
|             default = self._default
 | |
|         if annotation is MISSING:
 | |
|             annotation = self._annotation
 | |
|         if description is MISSING:
 | |
|             description = self._description
 | |
|         if displayed_default is MISSING:
 | |
|             displayed_default = self._displayed_default
 | |
| 
 | |
|         return self.__class__(
 | |
|             name=name,
 | |
|             kind=kind,
 | |
|             default=default,
 | |
|             annotation=annotation,
 | |
|             description=description,
 | |
|             displayed_default=displayed_default,
 | |
|         )
 | |
| 
 | |
|     if not TYPE_CHECKING:  # this is to prevent anything breaking if inspect internals change
 | |
|         name = _gen_property('name')
 | |
|         kind = _gen_property('kind')
 | |
|         default = _gen_property('default')
 | |
|         annotation = _gen_property('annotation')
 | |
| 
 | |
|     @property
 | |
|     def required(self) -> bool:
 | |
|         """:class:`bool`: Whether this parameter is required."""
 | |
|         return self.default is empty
 | |
| 
 | |
|     @property
 | |
|     def converter(self) -> Any:
 | |
|         """The converter that should be used for this parameter."""
 | |
|         if self.annotation is empty:
 | |
|             return type(self.default) if self.default not in (empty, None) else str
 | |
| 
 | |
|         return self.annotation
 | |
| 
 | |
|     @property
 | |
|     def description(self) -> Optional[str]:
 | |
|         """Optional[:class:`str`]: The description of this parameter."""
 | |
|         return self._description if self._description is not empty else None
 | |
| 
 | |
|     @property
 | |
|     def displayed_default(self) -> Optional[str]:
 | |
|         """Optional[:class:`str`]: The displayed default in :class:`Command.signature`."""
 | |
|         if self._displayed_default is not empty:
 | |
|             return self._displayed_default
 | |
| 
 | |
|         return None if self.required else str(self.default)
 | |
| 
 | |
|     async def get_default(self, ctx: Context[Any]) -> Any:
 | |
|         """|coro|
 | |
| 
 | |
|         Gets this parameter's default value.
 | |
| 
 | |
|         Parameters
 | |
|         ----------
 | |
|         ctx: :class:`Context`
 | |
|             The invocation context that is used to get the default argument.
 | |
|         """
 | |
|         # pre-condition: required is False
 | |
|         if callable(self.default):
 | |
|             return await maybe_coroutine(self.default, ctx)  # type: ignore
 | |
|         return self.default
 | |
| 
 | |
| 
 | |
| def parameter(
 | |
|     *,
 | |
|     converter: Any = empty,
 | |
|     default: Any = empty,
 | |
|     description: str = empty,
 | |
|     displayed_default: str = empty,
 | |
| ) -> Any:
 | |
|     r"""parameter(\*, converter=..., default=..., description=..., displayed_default=...)
 | |
| 
 | |
|     A way to assign custom metadata for a :class:`Command`\'s parameter.
 | |
| 
 | |
|     .. versionadded:: 2.0
 | |
| 
 | |
|     Examples
 | |
|     --------
 | |
|     A custom default can be used to have late binding behaviour.
 | |
| 
 | |
|     .. code-block:: python3
 | |
| 
 | |
|         @bot.command()
 | |
|         async def wave(ctx, to: discord.User = commands.parameter(default=lambda ctx: ctx.author)):
 | |
|             await ctx.send(f'Hello {to.mention} :wave:')
 | |
| 
 | |
|     Parameters
 | |
|     ----------
 | |
|     converter: Any
 | |
|         The converter to use for this parameter, this replaces the annotation at runtime which is transparent to type checkers.
 | |
|     default: Any
 | |
|         The default value for the parameter, if this is a :term:`callable` or a |coroutine_link|_ it is called with a
 | |
|         positional :class:`Context` argument.
 | |
|     description: :class:`str`
 | |
|         The description of this parameter.
 | |
|     displayed_default: :class:`str`
 | |
|         The displayed default in :attr:`Command.signature`.
 | |
|     """
 | |
|     return Parameter(
 | |
|         name='empty',
 | |
|         kind=inspect.Parameter.POSITIONAL_OR_KEYWORD,
 | |
|         annotation=converter,
 | |
|         default=default,
 | |
|         description=description,
 | |
|         displayed_default=displayed_default,
 | |
|     )
 | |
| 
 | |
| 
 | |
| class ParameterAlias(Protocol):
 | |
|     def __call__(
 | |
|         self,
 | |
|         *,
 | |
|         converter: Any = empty,
 | |
|         default: Any = empty,
 | |
|         description: str = empty,
 | |
|         displayed_default: str = empty,
 | |
|     ) -> Any:
 | |
|         ...
 | |
| 
 | |
| 
 | |
| param: ParameterAlias = parameter
 | |
| r"""param(\*, converter=..., default=..., description=..., displayed_default=...)
 | |
| 
 | |
| An alias for :func:`parameter`.
 | |
| 
 | |
| .. versionadded:: 2.0
 | |
| """
 | |
| 
 | |
| # some handy defaults
 | |
| Author = parameter(
 | |
|     default=attrgetter('author'),
 | |
|     displayed_default='<you>',
 | |
|     converter=Union[Member, User],
 | |
| )
 | |
| Author._fallback = True
 | |
| 
 | |
| CurrentChannel = parameter(
 | |
|     default=attrgetter('channel'),
 | |
|     displayed_default='<this channel>',
 | |
|     converter=Union[TextChannel, DMChannel, Thread, VoiceChannel],
 | |
| )
 | |
| CurrentChannel._fallback = True
 | |
| 
 | |
| 
 | |
| def default_guild(ctx: Context[Any]) -> Guild:
 | |
|     if ctx.guild is not None:
 | |
|         return ctx.guild
 | |
|     raise NoPrivateMessage()
 | |
| 
 | |
| 
 | |
| CurrentGuild = parameter(
 | |
|     default=default_guild,
 | |
|     displayed_default='<this server>',
 | |
|     converter=GuildConverter,
 | |
| )
 | |
| 
 | |
| 
 | |
| class Signature(inspect.Signature):
 | |
|     _parameter_cls = Parameter
 | |
|     parameters: OrderedDict[str, Parameter]
 |