Currently this generates an error:
from typing import Any
class C[*Ts, S]:
pass
a = C[int, str]()
a = C[*tuple[Any, ...], str]() # Error
This is the output:
error: Incompatible types in assignment (expression has type "C[*tuple[Any, ...], str]", variable has type "C[int, str]")
However, this doesn't generate an error:
from typing import Any
class C[*Ts, S]:
pass
a = C[int, str]()
a = C[*tuple[Any, ...]]() # No error
I'd argue that the first example shouldn't generate an error either. One possible rule would be to allow this as long as some substitution of the *tuple[Any, ...] part would match the target type.
We could possibly also generalize the matching of unknown-length TypeVarTuple type arguments when the unknown-length part has a non-Any tuple item type. This could reduce apparent false positives. Example:
from typing import Any
class C[*Ts, S]:
pass
a = C[int, int]()
a = C[*tuple[int, ...]]() # No error?
This would only change the behavior of variadic generics. Variable-length tuples would continue to behave as they behave currently, i.e. tuple[int, ...] wouldn't be assignable to tuple[int, int]. The tuple behavior has been around for a very long time, so it doesn't make sense to change it, but variadic generics are a relatively new and untested feature.
The primary motivation is help with NumPy and libraries that provide multidimensional array-like types.
Proper subtype checks should continue to work as they work currently, since we can't safely simplify say C[int] | C[*tuple[Any, ...]].
cc @ilevkivskyi who I chatted about this recently
Currently this generates an error:
This is the output:
However, this doesn't generate an error:
I'd argue that the first example shouldn't generate an error either. One possible rule would be to allow this as long as some substitution of the
*tuple[Any, ...]part would match the target type.We could possibly also generalize the matching of unknown-length TypeVarTuple type arguments when the unknown-length part has a non-
Anytuple item type. This could reduce apparent false positives. Example:This would only change the behavior of variadic generics. Variable-length tuples would continue to behave as they behave currently, i.e.
tuple[int, ...]wouldn't be assignable totuple[int, int]. The tuple behavior has been around for a very long time, so it doesn't make sense to change it, but variadic generics are a relatively new and untested feature.The primary motivation is help with NumPy and libraries that provide multidimensional array-like types.
Proper subtype checks should continue to work as they work currently, since we can't safely simplify say
C[int] | C[*tuple[Any, ...]].cc @ilevkivskyi who I chatted about this recently