mirror of
				https://github.com/astral-sh/ruff.git
				synced 2025-10-25 01:17:34 +00:00 
			
		
		
		
	 4ecf1d205a
			
		
	
	
		4ecf1d205a
		
			
		
	
	
	
	
		
			
			## Summary - Add support for the return types of `async` functions - Add type inference for `await` expressions - Add support for `async with` / async context managers - Add support for `yield from` expressions This PR is generally lacking proper error handling in some cases (e.g. illegal `__await__` attributes). I'm planning to work on this in a follow-up. part of https://github.com/astral-sh/ty/issues/151 closes https://github.com/astral-sh/ty/issues/736 ## Ecosystem There are a lot of true positives on `prefect` which look similar to: ```diff prefect (https://github.com/PrefectHQ/prefect) + src/integrations/prefect-aws/tests/workers/test_ecs_worker.py:406:12: error[unresolved-attribute] Type `str` has no attribute `status_code` ``` This is due to a wrong return type annotation [here](e926b8c4c1/src/integrations/prefect-aws/tests/workers/test_ecs_worker.py (L355-L391)). ```diff mitmproxy (https://github.com/mitmproxy/mitmproxy) + test/mitmproxy/addons/test_clientplayback.py:18:1: error[invalid-argument-type] Argument to function `asynccontextmanager` is incorrect: Expected `(...) -> AsyncIterator[Unknown]`, found `def tcp_server(handle_conn, **server_args) -> Unknown | tuple[str, int]` ``` [This](a4d794c59a/test/mitmproxy/addons/test_clientplayback.py (L18-L19)) is a true positive. That function should return `AsyncIterator[Address]`, not `Address`. I looked through almost all of the other new diagnostics and they all look like known problems or true positives. ## Typing conformance The typing conformance diff looks good. ## Test Plan New Markdown tests
		
			
				
	
	
	
	
		
			2.3 KiB
		
	
	
	
	
	
	
	
			
		
		
	
	
			2.3 KiB
		
	
	
	
	
	
	
	
async / await
Basic
async def retrieve() -> int:
    return 42
async def main():
    result = await retrieve()
    reveal_type(result)  # revealed: int
Generic async functions
from typing import TypeVar
T = TypeVar("T")
async def persist(x: T) -> T:
    return x
async def f(x: int):
    result = await persist(x)
    reveal_type(result)  # revealed: int
Use cases
Future
import asyncio
import concurrent.futures
def blocking_function() -> int:
    return 42
async def main():
    loop = asyncio.get_event_loop()
    with concurrent.futures.ThreadPoolExecutor() as pool:
        result = await loop.run_in_executor(pool, blocking_function)
        # TODO: should be `int`
        reveal_type(result)  # revealed: Unknown
asyncio.Task
import asyncio
async def f() -> int:
    return 1
async def main():
    task = asyncio.create_task(f())
    result = await task
    # TODO: this should be `int`
    reveal_type(result)  # revealed: Unknown
asyncio.gather
import asyncio
async def task(name: str) -> int:
    return len(name)
async def main():
    (a, b) = await asyncio.gather(
        task("A"),
        task("B"),
    )
    # TODO: these should be `int`
    reveal_type(a)  # revealed: Unknown
    reveal_type(b)  # revealed: Unknown
Under the hood
[environment]
python-version = "3.12"  # Use 3.12 to be able to use PEP 695 generics
Let's look at the example from the beginning again:
async def retrieve() -> int:
    return 42
When we look at the signature of this function, we see that it actually returns a CoroutineType:
reveal_type(retrieve)  # revealed: def retrieve() -> CoroutineType[Any, Any, int]
The expression await retrieve() desugars into a call to the __await__ dunder method on the
CoroutineType object, followed by a yield from. Let's first see the return type of __await__:
reveal_type(retrieve().__await__())  # revealed: Generator[Any, None, int]
We can see that this returns a Generator that yields Any, and eventually returns int. For the
final type of the await expression, we retrieve that third argument of the Generator type:
from typing import Generator
def _():
    result = yield from retrieve().__await__()
    reveal_type(result)  # revealed: int