crates/ty_python_semantic/resources/mdtest/generics/pep695/callables.md
[environment]
python-version = "3.12"
Many items that are callable can also be generic. Generic functions are the most obvious example:
from typing import Callable
from ty_extensions import generic_context
def identity[T](t: T) -> T:
return t
# revealed: ty_extensions.GenericContext[T@identity]
reveal_type(generic_context(identity))
# revealed: Literal[1]
reveal_type(identity(1))
def identity2[**P, T](c: Callable[P, T]) -> Callable[P, T]:
return c
# revealed: ty_extensions.GenericContext[P@identity2, T@identity2]
reveal_type(generic_context(identity2))
# revealed: [T](t: T) -> T
reveal_type(identity2(identity))
Generic classes are another example, since you invoke the class to instantiate it:
class C[T]:
t: T # invariant
def __init__(self, t: T) -> None: ...
# revealed: ty_extensions.GenericContext[T@C]
reveal_type(generic_context(C))
# revealed: C[int]
reveal_type(C(1))
When we coerce a generic callable into a Callable type, it remembers that it is generic:
from ty_extensions import into_regular_callable
# revealed: [T](t: T) -> T
reveal_type(into_regular_callable(identity))
# revealed: ty_extensions.GenericContext[T@identity]
reveal_type(generic_context(into_regular_callable(identity)))
# revealed: Literal[1]
reveal_type(into_regular_callable(identity)(1))
# revealed: [**P, T](c: (**P) -> T) -> ((**P) -> T)
reveal_type(into_regular_callable(identity2))
# revealed: ty_extensions.GenericContext[P@identity2, T@identity2]
reveal_type(generic_context(into_regular_callable(identity2)))
# revealed: [T](t: T) -> T
reveal_type(into_regular_callable(identity2)(identity))
# revealed: [T](t: T) -> C[T]
reveal_type(into_regular_callable(C))
# revealed: ty_extensions.GenericContext[T@C]
reveal_type(generic_context(into_regular_callable(C)))
# revealed: C[int]
reveal_type(into_regular_callable(C)(1))
Callable: type aliasesThe easiest way to refer to a generic Callable type directly is via a type alias:
from typing import Callable
from ty_extensions import generic_context
type IdentityCallable[T] = Callable[[T], T]
def decorator_factory[T]() -> IdentityCallable[T]:
def decorator[T](fn: T) -> T:
return fn
# revealed: ty_extensions.GenericContext[T@decorator]
reveal_type(generic_context(decorator))
return decorator
# Note that `decorator_factory` returns a generic callable, but is not itself generic!
# revealed: None
reveal_type(generic_context(decorator_factory))
# revealed: [T'return](T'return, /) -> T'return
reveal_type(decorator_factory())
# revealed: ty_extensions.GenericContext[T'return@decorator_factory]
reveal_type(generic_context(decorator_factory()))
# revealed: Literal[1]
reveal_type(decorator_factory()(1))
Callable with paramspecs: type aliasesThe same pattern holds if the callable involves a paramspec.
from typing import Callable
from ty_extensions import generic_context
type IdentityCallable[**P, T] = Callable[[Callable[P, T]], Callable[P, T]]
def decorator_factory[**P, T]() -> IdentityCallable[P, T]:
def decorator[**P, T](fn: Callable[P, T]) -> Callable[P, T]:
return fn
# revealed: ty_extensions.GenericContext[P@decorator, T@decorator]
reveal_type(generic_context(decorator))
return decorator
# Note that `decorator_factory` returns a generic callable, but is not itself generic!
# revealed: None
reveal_type(generic_context(decorator_factory))
def identity[T](t: T) -> T:
return t
# revealed: [**P'return, T'return]((**P'return) -> T'return, /) -> ((**P'return) -> T'return)
reveal_type(decorator_factory())
# revealed: ty_extensions.GenericContext[P'return@decorator_factory, T'return@decorator_factory]
reveal_type(generic_context(decorator_factory()))
# revealed: [T](t: T) -> T
reveal_type(decorator_factory()(identity))
# revealed: Literal[1]
reveal_type(decorator_factory()(identity)(1))
Callable: function return valuesYou can also return a generic Callable from a function. If a typevar only appears inside of
Callable, and only in return type position, then we treat the callable as generic, not the
function, just like above.
NOTE: This is one place where the PEP-695 syntax is misleading! It looks like decorator_factory
is generic, since it contains a [T] binding context. However, we still notice that the only use
of T in the signature is in the return type, inside of a Callable — and so it is the returned
callable that is generic, not the function.
from typing import Callable
from ty_extensions import generic_context
def decorator_factory[T]() -> Callable[[T], T]:
def decorator[T](fn: T) -> T:
return fn
# revealed: ty_extensions.GenericContext[T@decorator]
reveal_type(generic_context(decorator))
return decorator
# Note that `decorator_factory` returns a generic callable, but is not itself generic!
# revealed: None
reveal_type(generic_context(decorator_factory))
# revealed: [T'return](T'return, /) -> T'return
reveal_type(decorator_factory())
# revealed: ty_extensions.GenericContext[T'return@decorator_factory]
reveal_type(generic_context(decorator_factory()))
# revealed: Literal[1]
reveal_type(decorator_factory()(1))
If the typevar also appears in a parameter, it is the function that is generic, and the returned
Callable is not:
def outside_callable[T](t: T) -> Callable[[T], T]:
raise NotImplementedError
# revealed: ty_extensions.GenericContext[T@outside_callable]
reveal_type(generic_context(outside_callable))
# revealed: (int, /) -> int
reveal_type(outside_callable(1))
# revealed: None
reveal_type(generic_context(outside_callable(1)))
# error: [invalid-argument-type]
outside_callable(1)("string")
Callable with paramspecs: function return valuesThe same pattern holds if the callable involves a paramspec.
from typing import Callable
from ty_extensions import generic_context
def decorator_factory[**P, T]() -> Callable[[Callable[P, T]], Callable[P, T]]:
def decorator[**P, T](fn: Callable[P, T]) -> Callable[P, T]:
return fn
# revealed: ty_extensions.GenericContext[P@decorator, T@decorator]
reveal_type(generic_context(decorator))
return decorator
# Note that `decorator_factory` returns a generic callable, but is not itself generic!
# revealed: None
reveal_type(generic_context(decorator_factory))
def identity[T](t: T) -> T:
return t
# revealed: [**P'return, T'return]((**P'return) -> T'return, /) -> ((**P'return) -> T'return)
reveal_type(decorator_factory())
# revealed: ty_extensions.GenericContext[P'return@decorator_factory, T'return@decorator_factory]
reveal_type(generic_context(decorator_factory()))
# revealed: [T](t: T) -> T
reveal_type(decorator_factory()(identity))
# revealed: Literal[1]
reveal_type(decorator_factory()(identity)(1))
If the typevar also appears in a parameter, it is the function that is generic, and the returned
Callable is not:
def outside_callable[**P, T](func: Callable[P, T]) -> Callable[P, T]:
raise NotImplementedError
# revealed: ty_extensions.GenericContext[P@outside_callable, T@outside_callable]
reveal_type(generic_context(outside_callable))
def int_identity(x: int) -> int:
return x
# revealed: (x: int) -> int
reveal_type(outside_callable(int_identity))
# revealed: None
reveal_type(generic_context(outside_callable(int_identity)))
# error: [invalid-argument-type]
outside_callable(int_identity)("string")
Callable argumentAn overloaded callable should be assignable to a non-overloaded callable type when the overload set as a whole is compatible with the target callable.
The type variable should be inferred from the first matching overload, rather than unioning parameter types across all overloads (which would create an unsatisfiable expected type for contravariant type variables).
from typing import Callable, overload
def accepts_callable[T](converter: Callable[[T], None]) -> T:
raise NotImplementedError
@overload
def f(val: str) -> None: ...
@overload
def f(val: bytes) -> None: ...
def f(val: str | bytes) -> None:
pass
reveal_type(accepts_callable(f)) # revealed: str | bytes
When T is constrained to a union by other arguments, the overloaded callable must still be treated
as a whole to satisfy Callable[[T], T].
from typing import Callable, overload
def apply_twice[T](converter: Callable[[T], T], left: T, right: T) -> tuple[T, T]:
return converter(left), converter(right)
@overload
def f(val: int) -> int: ...
@overload
def f(val: str) -> str: ...
def f(val: int | str) -> int | str:
return val
x: int | str = 1
y: int | str = "a"
result = apply_twice(f, x, y)
# revealed: tuple[int | str, int | str]
reveal_type(result)
An overloaded callable returned from a generic callable factory should still be assignable to the declared generic callable return type.
from collections.abc import Callable, Coroutine
from typing import Any, overload
def singleton[S](flag: bool = False) -> Callable[[Callable[[int], S]], Callable[[int], S]]:
@overload
def wrapper[T](func: Callable[[int], Coroutine[Any, Any, T]]) -> Callable[[int], Coroutine[Any, Any, T]]: ...
@overload
def wrapper[U](func: Callable[[int], U]) -> Callable[[int], U]: ...
def wrapper[T, U](func: Callable[[int], Coroutine[Any, Any, T] | U]) -> Callable[[int], Coroutine[Any, Any, T] | U]:
return func
return wrapper
Reduced regression lock for a SymPy overload/protocol shape that can panic in the overload-assignability path.
from __future__ import annotations
from sympy.polys.compatibility import Domain, IPolys
from typing import overload
class DefaultPrinting:
pass
class PolyRing[T](DefaultPrinting, IPolys[T]):
symbols: tuple[object, ...]
domain: Domain[T]
def clone(
self,
symbols: object | None = None,
domain: object | None = None,
order: object | None = None,
) -> PolyRing[T]:
return self
@overload
def __getitem__(self, key: int) -> PolyRing[T]: ...
@overload
def __getitem__(self, key: slice) -> PolyRing[T] | Domain[T]: ...
def __getitem__(self, key: slice | int) -> PolyRing[T] | Domain[T]:
symbols = self.symbols[key]
if not symbols:
return self.domain
return self.clone(symbols=symbols)
def takes_ring(x: PolyRing[int]) -> None:
reveal_type(x[0]) # revealed: PolyRing[int]
reveal_type(x[:]) # revealed: PolyRing[int] | Domain[int]
sympy/polys/compatibility.pyi:
from __future__ import annotations
from typing import Protocol, overload
class Domain[T]: ...
class IPolys[T](Protocol):
@overload
def clone(
self,
symbols: object | None = None,
domain: None = None,
order: None = None,
) -> IPolys[T]: ...
@overload
def clone[S](
self,
symbols: object | None = None,
*,
domain: Domain[S],
order: None = None,
) -> IPolys[S]: ...
@overload
def __getitem__(self, key: int) -> IPolys[T]: ...
@overload
def __getitem__(self, key: slice) -> IPolys[T] | Domain[T]: ...