mirror of
https://github.com/Rapptz/discord.py.git
synced 2025-10-24 18:13:00 +00:00
Fix nested Annotated calls not resolving
This commit is contained in:
3
.github/workflows/test.yml
vendored
3
.github/workflows/test.yml
vendored
@@ -26,8 +26,7 @@ jobs:
|
|||||||
|
|
||||||
- name: Install dependencies
|
- name: Install dependencies
|
||||||
run: |
|
run: |
|
||||||
python -m pip install --upgrade pip setuptools wheel "coverage[toml]" pytest pytest-asyncio pytest-cov pytest-mock
|
python -m pip install -e .[test]
|
||||||
pip install -U -r requirements.txt
|
|
||||||
|
|
||||||
- name: Run tests
|
- name: Run tests
|
||||||
shell: bash
|
shell: bash
|
||||||
|
@@ -750,9 +750,6 @@ def get_supported_annotation(
|
|||||||
if isinstance(annotation, Transformer):
|
if isinstance(annotation, Transformer):
|
||||||
return (annotation, MISSING, False)
|
return (annotation, MISSING, False)
|
||||||
|
|
||||||
if hasattr(annotation, '__metadata__'):
|
|
||||||
return get_supported_annotation(annotation.__metadata__[0])
|
|
||||||
|
|
||||||
if inspect.isclass(annotation):
|
if inspect.isclass(annotation):
|
||||||
if issubclass(annotation, Transformer):
|
if issubclass(annotation, Transformer):
|
||||||
return (annotation(), MISSING, False)
|
return (annotation(), MISSING, False)
|
||||||
|
@@ -160,12 +160,6 @@ def get_signature_parameters(
|
|||||||
if annotation is Greedy:
|
if annotation is Greedy:
|
||||||
raise TypeError('Unparameterized Greedy[...] is disallowed in signature.')
|
raise TypeError('Unparameterized Greedy[...] is disallowed in signature.')
|
||||||
|
|
||||||
if hasattr(annotation, '__metadata__'):
|
|
||||||
# Annotated[X, Y] can access Y via __metadata__
|
|
||||||
metadata = annotation.__metadata__
|
|
||||||
if len(metadata) >= 1:
|
|
||||||
annotation = metadata[0]
|
|
||||||
|
|
||||||
params[name] = parameter.replace(annotation=annotation)
|
params[name] = parameter.replace(annotation=annotation)
|
||||||
|
|
||||||
return params
|
return params
|
||||||
|
@@ -184,11 +184,6 @@ def get_flags(namespace: Dict[str, Any], globals: Dict[str, Any], locals: Dict[s
|
|||||||
flag.name = name
|
flag.name = name
|
||||||
|
|
||||||
annotation = flag.annotation = resolve_annotation(flag.annotation, globals, locals, cache)
|
annotation = flag.annotation = resolve_annotation(flag.annotation, globals, locals, cache)
|
||||||
if hasattr(annotation, '__metadata__'):
|
|
||||||
# Annotated[X, Y] can access Y via __metadata__
|
|
||||||
metadata = annotation.__metadata__
|
|
||||||
if len(metadata) >= 1:
|
|
||||||
annotation = flag.annotation = metadata[0]
|
|
||||||
|
|
||||||
if flag.default is MISSING and hasattr(annotation, '__commands_is_flag__') and annotation._can_be_constructible():
|
if flag.default is MISSING and hasattr(annotation, '__commands_is_flag__') and annotation._can_be_constructible():
|
||||||
flag.default = annotation._construct_default
|
flag.default = annotation._construct_default
|
||||||
|
@@ -1062,6 +1062,11 @@ def evaluate_annotation(
|
|||||||
cache[tp] = evaluated
|
cache[tp] = evaluated
|
||||||
return evaluated
|
return evaluated
|
||||||
|
|
||||||
|
if hasattr(tp, '__metadata__'):
|
||||||
|
# Annotated[X, Y] can access Y via __metadata__
|
||||||
|
metadata = tp.__metadata__[0]
|
||||||
|
return evaluate_annotation(metadata, globals, locals, cache)
|
||||||
|
|
||||||
if hasattr(tp, '__args__'):
|
if hasattr(tp, '__args__'):
|
||||||
implicit_str = True
|
implicit_str = True
|
||||||
is_literal = False
|
is_literal = False
|
||||||
|
5
setup.py
5
setup.py
@@ -39,7 +39,7 @@ extras_require = {
|
|||||||
'sphinx==4.4.0',
|
'sphinx==4.4.0',
|
||||||
'sphinxcontrib_trio==1.1.2',
|
'sphinxcontrib_trio==1.1.2',
|
||||||
'sphinxcontrib-websupport',
|
'sphinxcontrib-websupport',
|
||||||
'typing-extensions',
|
'typing-extensions>=4.3,<5',
|
||||||
],
|
],
|
||||||
'speed': [
|
'speed': [
|
||||||
'orjson>=3.5.4',
|
'orjson>=3.5.4',
|
||||||
@@ -52,7 +52,8 @@ extras_require = {
|
|||||||
'pytest',
|
'pytest',
|
||||||
'pytest-asyncio',
|
'pytest-asyncio',
|
||||||
'pytest-cov',
|
'pytest-cov',
|
||||||
'pytest-mock'
|
'pytest-mock',
|
||||||
|
'typing-extensions>=4.3,<5',
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
|
||||||
|
61
tests/test_annotated_annotation.py
Normal file
61
tests/test_annotated_annotation.py
Normal file
@@ -0,0 +1,61 @@
|
|||||||
|
"""
|
||||||
|
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 typing import Optional
|
||||||
|
from typing_extensions import Annotated
|
||||||
|
|
||||||
|
import discord
|
||||||
|
from discord import app_commands
|
||||||
|
from discord.ext import commands
|
||||||
|
|
||||||
|
import pytest
|
||||||
|
|
||||||
|
def test_annotated_annotation():
|
||||||
|
# can't exactly test if the parameter is the same, so just test if it raises something
|
||||||
|
@app_commands.command()
|
||||||
|
async def foo(interaction: discord.Interaction, param: Annotated[float, Optional[int]]):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
def to_hex(arg: str) -> int:
|
||||||
|
return int(arg, 16)
|
||||||
|
|
||||||
|
class Flag(commands.FlagConverter):
|
||||||
|
thing: Annotated[int, to_hex]
|
||||||
|
|
||||||
|
assert Flag.get_flags()['thing'].annotation == to_hex
|
||||||
|
|
||||||
|
@commands.command()
|
||||||
|
async def bar(ctx: commands.Context, param: Annotated[float, Optional[int]]):
|
||||||
|
pass
|
||||||
|
|
||||||
|
assert bar.clean_params['param'].annotation == Optional[int]
|
||||||
|
|
||||||
|
@commands.command()
|
||||||
|
async def nested(ctx: commands.Context, param: Optional[Annotated[str, int]]):
|
||||||
|
pass
|
||||||
|
|
||||||
|
assert nested.clean_params['param'].annotation == Optional[int]
|
||||||
|
|
Reference in New Issue
Block a user