KEMBAR78
Reject invalid ParamSpec locations by sterliakov · Pull Request #18278 · python/mypy · GitHub
Skip to content

Conversation

@sterliakov
Copy link
Collaborator

@sterliakov sterliakov commented Dec 11, 2024

Fixes #14832, fixes #13966, fixes #14622.

Still does not report error in #14777, I'll work separately on that.

Move all ParamSpec validity checking to typeanal.py. Stop treating P.args and P.kwargs as binding - only bare typevar makes it available in scope. Reject keyword arguments following P.args.

This also makes one more conformance test pass. Diff (against baseline on current master):

diff --git a/conformance/results/mypy/generics_paramspec_components.toml b/conformance/results/mypy/generics_paramspec_components.toml
index 66a386a..86ba0b0 100644
--- a/conformance/results/mypy/generics_paramspec_components.toml
+++ b/conformance/results/mypy/generics_paramspec_components.toml
@@ -8,10 +8,18 @@ Does not report error when keyword argument is specified between P.args and P.kw
 output = """
 generics_paramspec_components.py:17: error: Use "P.args" for variadic "*" parameter  [valid-type]
 generics_paramspec_components.py:17: error: Use "P.kwargs" for variadic "**" parameter  [valid-type]
+generics_paramspec_components.py:20: error: ParamSpec components are not allowed here  [valid-type]
 generics_paramspec_components.py:23: error: Use "P.kwargs" for variadic "**" parameter  [valid-type]
+generics_paramspec_components.py:26: error: ParamSpec must have "*args" typed as "P.args" and "**kwargs" typed as "P.kwargs"  [valid-type]
+generics_paramspec_components.py:30: error: ParamSpec "P" is unbound  [valid-type]
+generics_paramspec_components.py:35: error: ParamSpec components are not allowed here  [valid-type]
+generics_paramspec_components.py:36: error: ParamSpec components are not allowed here  [valid-type]
+generics_paramspec_components.py:38: error: ParamSpec must have "*args" typed as "P.args" and "**kwargs" typed as "P.kwargs"  [valid-type]
+generics_paramspec_components.py:41: error: ParamSpec must have "*args" typed as "P.args" and "**kwargs" typed as "P.kwargs"  [valid-type]
 generics_paramspec_components.py:49: error: Argument 1 has incompatible type "*P.kwargs"; expected "P.args"  [arg-type]
 generics_paramspec_components.py:49: error: Argument 2 has incompatible type "**P.args"; expected "P.kwargs"  [arg-type]
 generics_paramspec_components.py:51: error: Argument 1 has incompatible type "int"; expected "P.args"  [arg-type]
+generics_paramspec_components.py:60: error: Arguments not allowed after ParamSpec.args  [valid-type]
 generics_paramspec_components.py:70: error: Argument 1 has incompatible type "*P.args"; expected "int"  [arg-type]
 generics_paramspec_components.py:70: error: Argument 2 has incompatible type "int"; expected "P.args"  [arg-type]
 generics_paramspec_components.py:72: error: Argument 1 has incompatible type "*P.args"; expected "int"  [arg-type]
@@ -21,14 +29,6 @@ generics_paramspec_components.py:83: error: Argument 3 to "foo" has incompatible
 generics_paramspec_components.py:98: error: Argument 2 to "twice" has incompatible type "str"; expected "int"  [arg-type]
 generics_paramspec_components.py:98: error: Argument 3 to "twice" has incompatible type "int"; expected "str"  [arg-type]
 """
-conformance_automated = "Fail"
+conformance_automated = "Pass"
 errors_diff = """
-Line 20: Expected 1 errors
-Line 26: Expected 1 errors
-Line 30: Expected 1 errors
-Line 35: Expected 1 errors
-Line 36: Expected 1 errors
-Line 38: Expected 1 errors
-Line 41: Expected 1 errors
-Line 60: Expected 1 errors
 """
diff --git a/conformance/results/mypy/version.toml b/conformance/results/mypy/version.toml
index 3030c9f..42ef13b 100644
--- a/conformance/results/mypy/version.toml
+++ b/conformance/results/mypy/version.toml
@@ -1,2 +1,2 @@
-version = "mypy 1.14.0+dev.14974072c0a70f8ca29c17c740475187b800e714"
-test_duration = 8.4
+version = "mypy 1.14.0+dev.e580045ce62e3c8ccd17fb575f60ec7ede65c01b"
+test_duration = 8.2
diff --git a/conformance/results/results.html b/conformance/results/results.html
index e6ef533..1278a53 100644
--- a/conformance/results/results.html
+++ b/conformance/results/results.html
@@ -158,8 +158,8 @@
         </header>
         <div class="table_container"><table><tbody>
 <tr><th class="col1">&nbsp;</th>
-<th class='tc-header'><div class='tc-name'>mypy 1.14.0+dev.14974072c0a70f8ca29c17c740475187b800e714</div>
-<div class='tc-time'>8.4sec</div>
+<th class='tc-header'><div class='tc-name'>mypy 1.14.0+dev.e580045ce62e3c8ccd17fb575f60ec7ede65c01b</div>
+<div class='tc-time'>8.2sec</div>
 </th>
 </tr>
 <tr><th class="column" colspan="2">

@github-actions

This comment has been minimized.

@github-actions

This comment has been minimized.

@sterliakov sterliakov marked this pull request as ready for review December 11, 2024 05:06
Copy link
Collaborator

@JukkaL JukkaL left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks, I'm glad to see several bugs fixed. Looks good!

@JukkaL
Copy link
Collaborator

JukkaL commented Dec 20, 2024

Actually, the home-assistant error might be false positive. Can you look at it?

@sterliakov
Copy link
Collaborator Author

Amazing. I wasn't able to reproduce the problem on a local clone, and indeed it was apparently a glitch: another primer run did not produce an error there.

@github-actions

This comment has been minimized.

@cdce8p
Copy link
Collaborator

cdce8p commented Dec 20, 2024

Actually, the home-assistant error might be false positive. Can you look at it?

Amazing. I wasn't able to reproduce the problem on a local clone, and indeed it was apparently a glitch: another primer run did not produce an error there.

No it wasn't a glitch. I just fixed it already after seeing your PR 😉
home-assistant/core#133044

_P wasn't bound by the outer function scope.

@sterliakov
Copy link
Collaborator Author

Ough! I was moderately certain that I reviewed primer hits and found them to be correct, but did not expect a fix in between... Please comment next time you do that!:)

Copy link
Collaborator

@hauntsaninja hauntsaninja left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This looks awesome! Makes things cleaner too.

I had one small nit regarding a test case

T = TypeVar("T")

def smoke_testable(*args: P.args, **kwargs: P.kwargs) -> Callable[[Callable[P, T]], Type[T]]:
def smoke_testable(*args: P.args, **kwargs: P.kwargs) -> Callable[[Callable[P, T]], Type[T]]: # E: ParamSpec "P" is unbound
Copy link
Collaborator

@hauntsaninja hauntsaninja Dec 30, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

this is functionality that mypy is losing, right? (you can see what it's going for in the interesting OtherClass example)

i think that's fine (and seems fine looking at the new primer warnings), but maybe we should clarify this test case? e.g. at least the comments are out of date

Copy link
Collaborator Author

@sterliakov sterliakov Jan 2, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks! I've removed this test. This definition is not allowed by PEP612:

These “properties” can only be used as the annotated types for *args and **kwargs, accessed from a ParamSpec already in scope.

*args and **kwargs do not add a ParamSpec to the scope, so the test has never been conformant. IMO this is an extremely rare case: "given a set of arguments, produce a callable depending on those". I think dropping this implementation-defined extension to prevent silently accepting and misinterpreting invalid typing constructs is sensible.

@github-actions
Copy link
Contributor

github-actions bot commented Jan 2, 2025

Diff from mypy_primer, showing the effect of this PR on open source code:

altair (https://github.com/vega/altair)
+ altair/theme.py:251: error: ParamSpec "P" is unbound  [valid-type]

prefect (https://github.com/PrefectHQ/prefect)
+ src/prefect/flows.py:1260: error: Arguments not allowed after ParamSpec.args  [valid-type]
+ src/prefect/flows.py:1269: error: Arguments not allowed after ParamSpec.args  [valid-type]
+ src/prefect/flows.py:1277: error: Arguments not allowed after ParamSpec.args  [valid-type]
+ src/prefect/tasks.py:957: error: Arguments not allowed after ParamSpec.args  [valid-type]
+ src/prefect/tasks.py:969: error: Arguments not allowed after ParamSpec.args  [valid-type]
+ src/prefect/tasks.py:979: error: Arguments not allowed after ParamSpec.args  [valid-type]
+ src/prefect/tasks.py:989: error: Arguments not allowed after ParamSpec.args  [valid-type]
+ src/prefect/tasks.py:998: error: Arguments not allowed after ParamSpec.args  [valid-type]
+ src/prefect/tasks.py:1043: error: Arguments not allowed after ParamSpec.args  [valid-type]
+ src/prefect/tasks.py:1053: error: Arguments not allowed after ParamSpec.args  [valid-type]
+ src/prefect/tasks.py:1063: error: Arguments not allowed after ParamSpec.args  [valid-type]
+ src/prefect/tasks.py:1073: error: Arguments not allowed after ParamSpec.args  [valid-type]

pwndbg (https://github.com/pwndbg/pwndbg)
+ pwndbg/gdblib/events.py: note: In function "connect":
+ pwndbg/gdblib/events.py:277: error: ParamSpec "P" is unbound  [valid-type]

@sterliakov
Copy link
Collaborator Author

In the meantime prefect fixed one occurrence of arg between *args and **kwargs and added a few more invalid functions of the same kind:)

@hauntsaninja
Copy link
Collaborator

Thank you!

@hauntsaninja hauntsaninja merged commit 025642b into python:master Jan 2, 2025
18 checks passed
@dangotbanned
Copy link

Diff from mypy_primer, showing the effect of this PR on open source code:

altair (https://github.com/vega/altair)
+ altair/theme.py:251: error: ParamSpec "P" is unbound  [valid-type] 

Hey (https://github.com/vega/altair) maintainer here

Any advice on how to resolve this?

@hauntsaninja
Copy link
Collaborator

hauntsaninja commented Feb 5, 2025

Hello hello! There's nothing that puts P into scope / nothing that tells a type checker what P is meant to match.

If you changed def decorate(func: Plugin[ThemeConfig], /) -> Plugin[ThemeConfig]: to def decorate(func: Callable[P, ThemeConfig], /) -> Callable[P, ThemeConfig]: that would do the job and actually provide better type checking to callers. (Looks like Plugin is basically just an alias for Callable that loses track of parameter types)

It's also unclear to me why you need an inner function at all. Seems like you could just return func from decorate and not bother with wrapper. This would also be a fix.

@dangotbanned
Copy link

Thanks @hauntsaninja for the quick response and help

@notypecheck
Copy link

Hello, I'm facing issue with this PR since updating mypy:

from typing import ParamSpec, Generic

P = ParamSpec("P")


class FunctionArgs(Generic[P]):
    args: P.args
    kwargs: P.kwargs
def something(func: Callable[P, Any]) -> FunctionArgs[P]: ...
$ mypy code.py
code.py:7:11: error: ParamSpec components are not allowed here  [valid-type]
        args: P.args
              ^
code.py:8:13: error: ParamSpec components are not allowed here  [valid-type]
        kwargs: P.kwargs

I'm not sure if it's "compatible" with current typing PEPs/Spec, but this seems like a valid use case to me.

@sterliakov
Copy link
Collaborator Author

sterliakov commented Feb 5, 2025 via email

@notypecheck
Copy link

I see, so main issue is not being able to statically type it since function could be called differently, thanks.
I guess my actual use case is a bit different?

@dataclasses.dataclass(slots=True, kw_only=True, frozen=True)
class TaskInstance(Generic[P, TResult]):
    task: TaskDefinition[P, TResult]
    args: P.args
    kwargs: P.kwargs


@dataclasses.dataclass(kw_only=True)
class TaskDefinition(Generic[P, TResult]):
    func: Callable[P, Awaitable[TResult]]

    def __call__(self, *args: P.args, **kwargs: P.kwargs) -> TaskInstance[P, TResult]:
        return TaskInstance(
            task=self,
            args=args,
            kwargs=kwargs,
        )

This seems ok since we know which arguments were used when creating instance of TaskInstance, but I guess if it's currently not supported or isn't planned I'll just type it as tuple[object, ...], dict[str, object] or something like that 😅

Joshix-1 added a commit to Joshix-1/typed_stream that referenced this pull request Feb 12, 2025
SEE: python/mypy#18278
The added Any doesn't change the behaviour of mypy, it was Any before as well.
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Labels

Projects

None yet

6 participants