Skip to content

Commit 9ffb9dd

Browse files
Fix crash when passing too many type arguments to generic base class accepting single ParamSpec (python#17770)
Fixes python#17765 The offender for this crash appears to be this snippet: https://github.com/python/mypy/blob/72c413d2352da5ce1433ef241faca8f40fa1fe27/mypy/semanal.py#L5905-L5910 This branch triggers when applying type args to a type that is generic with respect to a single `ParamSpec`. It allows double brackets to be omitted when providing a parameter specification by wrapping all of the provided type arguments into a single parameter specification argument (i.e. equating `Foo[int, int]` to `Foo[[int, int]]`). This wrapping occurs *unless*: * there is only a single type argument, and it resolves to `Any` (e.g. `Foo[Any]`) * **there is only a single type argument**, and it's a bracketed parameter specification or a `ParamSpec` (e.g. `Foo[[int, int]]`) The problem occurs when multiple type arguments provided and at least one of them is a bracketed parameter specification, as in `Foo[[int, int], str]`. Per the rules above, since there is more than 1 type argument, mypy attempts to wrap the arguments into a single parameter specification. This results in the attempted creation of a `Parameters` instance that contains another `Parameters` instance, which triggers this assert inside `Parameters.__init__`: https://github.com/python/mypy/blob/72c413d2352da5ce1433ef241faca8f40fa1fe27/mypy/types.py#L1634 I think a reasonable solution is to forgo wrapping the type arguments into a single `Parameters` if **any** of the provided type arguments are a `Parameters`/`ParamSpecType`. That is, don't transform `Foo[A1, A2, ...]` to `Foo[[A1, A2, ...]]` if any of `A1, A2, ...` are a parameter specification. This change brings the crash case inline with mypy's current behavior for a similar case: ```python # Current behavior P = ParamSpec("P") class C(Generic[P]): ... c: C[int, [int, str], str] # E: Nested parameter specifications are not allowed ``` Before this change: ```python P = ParamSpec("P") class C(Generic[P]): ... class D(C[int, [int, str], str]): ... # !!! CRASH !!! ``` After this change: ```python P = ParamSpec("P") class C(Generic[P]): ... class D(C[int, [int, str], str]): ... # E: Nested parameter specifications are not allowed ````
1 parent 94109aa commit 9ffb9dd

File tree

2 files changed

+11
-3
lines changed

2 files changed

+11
-3
lines changed

mypy/semanal.py

+2-3
Original file line numberDiff line numberDiff line change
@@ -5926,9 +5926,8 @@ def analyze_type_application_args(self, expr: IndexExpr) -> list[Type] | None:
59265926

59275927
if has_param_spec and num_args == 1 and types:
59285928
first_arg = get_proper_type(types[0])
5929-
if not (
5930-
len(types) == 1 and isinstance(first_arg, (Parameters, ParamSpecType, AnyType))
5931-
):
5929+
single_any = len(types) == 1 and isinstance(first_arg, AnyType)
5930+
if not (single_any or any(isinstance(t, (Parameters, ParamSpecType)) for t in types)):
59325931
types = [Parameters(types, [ARG_POS] * len(types), [None] * len(types))]
59335932

59345933
return types

test-data/unit/check-parameter-specification.test

+9
Original file line numberDiff line numberDiff line change
@@ -1836,6 +1836,15 @@ c: C[int, [int, str], str] # E: Nested parameter specifications are not allowed
18361836
reveal_type(c) # N: Revealed type is "__main__.C[Any]"
18371837
[builtins fixtures/paramspec.pyi]
18381838

1839+
[case testParamSpecInheritNoCrashOnNested]
1840+
from typing import Generic
1841+
from typing_extensions import ParamSpec
1842+
1843+
P = ParamSpec("P")
1844+
class C(Generic[P]): ...
1845+
class D(C[int, [int, str], str]): ... # E: Nested parameter specifications are not allowed
1846+
[builtins fixtures/paramspec.pyi]
1847+
18391848
[case testParamSpecConcatenateSelfType]
18401849
from typing import Callable
18411850
from typing_extensions import ParamSpec, Concatenate

0 commit comments

Comments
 (0)