@ -31,6 +31,7 @@ from typing import (
ClassVar ,
Dict ,
List ,
Tuple ,
Type ,
TypeVar ,
Union ,
@ -45,6 +46,7 @@ from .core import Command, Group
from . errors import BadArgument , CommandRegistrationError , CommandError , HybridCommandError , ConversionError
from . converter import Converter , Range , Greedy , run_converters
from . parameters import Parameter
from . flags import is_flag , FlagConverter
from . cog import Cog
from . view import StringView
@ -163,7 +165,86 @@ def make_greedy_transformer(converter: Any, parameter: Parameter) -> Type[app_co
return type ( ' GreedyTransformer ' , ( app_commands . Transformer , ) , { ' transform ' : classmethod ( transform ) } )
def replace_parameters ( parameters : Dict [ str , Parameter ] , signature : inspect . Signature ) - > List [ inspect . Parameter ] :
def replace_parameter (
param : inspect . Parameter ,
converter : Any ,
callback : Callable [ . . . , Any ] ,
original : Parameter ,
mapping : Dict [ str , inspect . Parameter ] ,
) - > inspect . Parameter :
try :
# If it's a supported annotation (i.e. a transformer) just let it pass as-is.
app_commands . transformers . get_supported_annotation ( converter )
except TypeError :
# Fallback to see if the behaviour needs changing
origin = getattr ( converter , ' __origin__ ' , None )
args = getattr ( converter , ' __args__ ' , [ ] )
if isinstance ( converter , Range ) :
r = converter
param = param . replace ( annotation = app_commands . Range [ r . annotation , r . min , r . max ] ) # type: ignore
elif isinstance ( converter , Greedy ) :
# Greedy is "optional" in ext.commands
# However, in here, it probably makes sense to make it required.
# I'm unsure how to allow the user to choose right now.
inner = converter . converter
if inner is discord . Attachment :
raise TypeError ( ' discord.Attachment with Greedy is not supported in hybrid commands ' )
param = param . replace ( annotation = make_greedy_transformer ( inner , original ) )
elif is_flag ( converter ) :
callback . __hybrid_command_flag__ = ( param . name , converter )
descriptions = { }
renames = { }
for flag in converter . __commands_flags__ . values ( ) :
name = flag . attribute
flag_param = inspect . Parameter (
name = name ,
kind = param . kind ,
default = flag . default if flag . default is not MISSING else inspect . Parameter . empty ,
annotation = flag . annotation ,
)
pseudo = replace_parameter ( flag_param , flag . annotation , callback , original , mapping )
if name in mapping :
raise TypeError ( f ' { name !r} flag would shadow a pre-existing parameter ' )
if flag . description is not MISSING :
descriptions [ name ] = flag . description
if flag . name != flag . attribute :
renames [ name ] = flag . name
mapping [ name ] = pseudo
# Manually call the decorators
if descriptions :
app_commands . describe ( * * descriptions ) ( callback )
if renames :
app_commands . rename ( * * renames ) ( callback )
elif is_converter ( converter ) :
param = param . replace ( annotation = make_converter_transformer ( converter ) )
elif origin is Union :
if len ( args ) == 2 and args [ - 1 ] is _NoneType :
# Special case Optional[X] where X is a single type that can optionally be a converter
inner = args [ 0 ]
is_inner_tranformer = is_transformer ( inner )
if is_converter ( inner ) and not is_inner_tranformer :
param = param . replace ( annotation = Optional [ make_converter_transformer ( inner ) ] ) # type: ignore
else :
raise
elif origin :
# Unsupported typing.X annotation e.g. typing.Dict, typing.Tuple, typing.List, etc.
raise
elif callable ( converter ) and not inspect . isclass ( converter ) :
param_count = required_pos_arguments ( converter )
if param_count != 1 :
raise
param = param . replace ( annotation = make_callable_transformer ( converter ) )
return param
def replace_parameters (
parameters : Dict [ str , Parameter ] , callback : Callable [ . . . , Any ] , signature : inspect . Signature
) - > List [ inspect . Parameter ] :
# Need to convert commands.Parameter back to inspect.Parameter so this will be a bit ugly
params = signature . parameters . copy ( )
for name , parameter in parameters . items ( ) :
@ -171,41 +252,7 @@ def replace_parameters(parameters: Dict[str, Parameter], signature: inspect.Sign
# Parameter.converter properly infers from the default and has a str default
# This allows the actual signature to inherit this property
param = params [ name ] . replace ( annotation = converter )
try :
# If it's a supported annotation (i.e. a transformer) just let it pass as-is.
app_commands . transformers . get_supported_annotation ( converter )
except TypeError :
# Fallback to see if the behaviour needs changing
origin = getattr ( converter , ' __origin__ ' , None )
args = getattr ( converter , ' __args__ ' , [ ] )
if isinstance ( converter , Range ) :
r = converter
param = param . replace ( annotation = app_commands . Range [ r . annotation , r . min , r . max ] ) # type: ignore
elif isinstance ( converter , Greedy ) :
# Greedy is "optional" in ext.commands
# However, in here, it probably makes sense to make it required.
# I'm unsure how to allow the user to choose right now.
inner = converter . converter
if inner is discord . Attachment :
raise TypeError ( ' discord.Attachment with Greedy is not supported in hybrid commands ' )
param = param . replace ( annotation = make_greedy_transformer ( inner , parameter ) )
elif is_converter ( converter ) :
param = param . replace ( annotation = make_converter_transformer ( converter ) )
elif origin is Union :
if len ( args ) == 2 and args [ - 1 ] is _NoneType :
# Special case Optional[X] where X is a single type that can optionally be a converter
inner = args [ 0 ]
is_inner_tranformer = is_transformer ( inner )
if is_converter ( inner ) and not is_inner_tranformer :
param = param . replace ( annotation = Optional [ make_converter_transformer ( inner ) ] ) # type: ignore
else :
raise
elif callable ( converter ) and not inspect . isclass ( converter ) :
param_count = required_pos_arguments ( converter )
if param_count != 1 :
raise
param = param . replace ( annotation = make_callable_transformer ( converter ) )
param = replace_parameter ( param , converter , callback , parameter , params )
if parameter . default is not parameter . empty :
default = _CallableDefault ( parameter . default ) if callable ( parameter . default ) else parameter . default
@ -215,6 +262,11 @@ def replace_parameters(parameters: Dict[str, Parameter], signature: inspect.Sign
# If we're here, then then it hasn't been handled yet so it should be removed completely
param = param . replace ( default = parameter . empty )
# Flags are flattened out and thus don't get their parameter in the actual mapping
if hasattr ( converter , ' __commands_is_flag__ ' ) :
del params [ name ]
continue
params [ name ] = param
return list ( params . values ( ) )
@ -223,7 +275,7 @@ def replace_parameters(parameters: Dict[str, Parameter], signature: inspect.Sign
class HybridAppCommand ( discord . app_commands . Command [ CogT , P , T ] ) :
def __init__ ( self , wrapped : Command [ CogT , Any , T ] ) - > None :
signature = inspect . signature ( wrapped . callback )
params = replace_parameters ( wrapped . params , signature )
params = replace_parameters ( wrapped . params , wrapped . callback , signature )
wrapped . callback . __signature__ = signature . replace ( parameters = params )
try :
@ -237,6 +289,10 @@ class HybridAppCommand(discord.app_commands.Command[CogT, P, T]):
self . wrapped : Command [ CogT , Any , T ] = wrapped
self . binding : Optional [ CogT ] = wrapped . cog
# This technically means only one flag converter is supported
self . flag_converter : Optional [ Tuple [ str , Type [ FlagConverter ] ] ] = getattr (
wrapped . callback , ' __hybrid_command_flag__ ' , None
)
def _copy_with ( self , * * kwargs ) - > Self :
copy : Self = super ( ) . _copy_with ( * * kwargs ) # type: ignore
@ -269,6 +325,19 @@ class HybridAppCommand(discord.app_commands.Command[CogT, P, T]):
else :
transformed_values [ param . name ] = await param . transform ( interaction , value )
if self . flag_converter is not None :
param_name , flag_cls = self . flag_converter
flag = flag_cls . __new__ ( flag_cls )
for f in flag_cls . __commands_flags__ . values ( ) :
try :
value = transformed_values . pop ( f . attribute )
except KeyError :
raise app_commands . CommandSignatureMismatch ( self ) from None
else :
setattr ( flag , f . attribute , value )
transformed_values [ param_name ] = flag
return transformed_values
async def _check_can_run ( self , interaction : discord . Interaction ) - > bool :