Skip to content

parameter specification/type variable tuple variance#2215

Open
KotlinIsland wants to merge 7 commits into
python:mainfrom
KotlinIsland:paramspec-variance
Open

parameter specification/type variable tuple variance#2215
KotlinIsland wants to merge 7 commits into
python:mainfrom
KotlinIsland:paramspec-variance

Conversation

@KotlinIsland KotlinIsland force-pushed the paramspec-variance branch 3 times, most recently from 5f8e0cc to d9d82c9 Compare March 9, 2026 03:13
@srittau srittau added the topic: typing spec For improving the typing spec label Mar 9, 2026
@KotlinIsland KotlinIsland force-pushed the paramspec-variance branch 2 times, most recently from 81ba400 to 54ad5e1 Compare April 1, 2026 07:59

@JelleZijlstra JelleZijlstra left a comment

Copy link
Copy Markdown
Member

Choose a reason for hiding this comment

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

I like adding this if we can get it specified nicely, but this PR is not ready; the proposed test is incorrect.

Also, https://typing.python.org/en/latest/spec/generics.html#paramspec-variables still says variance on ParamSpec is unsupported; this should be updated.

I'd also like to see an implementation in at least one type checker, even if only as a draft PR, so we can be confident this is something that can be feasibly implemented.

Comment thread conformance/tests/generics_paramspec_semantics.py Outdated
Comment thread conformance/tests/generics_paramspec_semantics.py Outdated
@davidhalter

Copy link
Copy Markdown
Collaborator

I personally would also like to see tests added for param spec variance inference, since that would probably need to be supported as well.

@JelleZijlstra

Copy link
Copy Markdown
Member

The test cases do have variance inference, though as I noted some of the cases are wrong. But it would probably be useful to have a few more cases, and I'd recommend putting the tests for paramspec variance in their own file so we can track type checker support more precisely.

@KotlinIsland

KotlinIsland commented Apr 3, 2026

Copy link
Copy Markdown
Contributor Author

I'd also like to see an implementation in at least one type checker

I have wip support in PyCharm, but I could also add it to basedpyright

it was very straightforward to implement

@davidhalter

davidhalter commented Apr 3, 2026

Copy link
Copy Markdown
Collaborator

The test cases do have variance inference, though as I noted some of the cases are wrong. But it would probably be useful to have a few more cases, and I'd recommend putting the tests for paramspec variance in their own file so we can track type checker support more precisely.

Oh right, sorry I didn't see them, because I thought they would be in a different file. I'm also very much +1 on putting those tests into a different file (for example generics_paramspec_variance.py).

@KotlinIsland KotlinIsland force-pushed the paramspec-variance branch 2 times, most recently from 8f30004 to 117964e Compare April 7, 2026 01:29
@KotlinIsland KotlinIsland changed the title parameter specification variance parameter specification/type variable tuple variance Apr 7, 2026
@KotlinIsland

Copy link
Copy Markdown
Contributor Author

support has landed in pycharm and cpython

@JelleZijlstra

Copy link
Copy Markdown
Member

https://typing.python.org/en/latest/spec/generics.html#paramspec-variables stills says that we don't support variance in ParamSpec, that should be fixed in this PR.

@JelleZijlstra

Copy link
Copy Markdown
Member

Also can you open an issue on python/typing-council asking for a formal pronouncement?

@KotlinIsland

Copy link
Copy Markdown
Contributor Author

python/typing-council#59

@JelleZijlstra

Copy link
Copy Markdown
Member

Noticed some more issues:

  • generics_typevartuple_basic.py still has a line Ts1 = TypeVarTuple("Ts1", covariant=True) # E. But of course that's legal now. I'd delete the line and leave all variance testing to the new file generics_typevartuple_variance.py.
  • The description of the variance inference algorithm says "In the upper specialized class, specialize the target type parameter with an object instance". But that doesn't make sense for ParamSpec and TypeVarTuple. I think TypeVarTuple should instead be specialized with *tuple[object, ...]. For ParamSpec perhaps Callable[..., object] is right but not 100% sure.
  • The way the tests are written produces some incidental errors, like from ty generics_paramspec_variance.py:28:23: error[empty-body] Function always implicitly returns `None`, which is not assignable to return type `(...) -> Unknown` and mypy generics_typevartuple_variance.py:18: error: Missing return statement [empty-body]. Should add something like raise NotImplementedError.

@KotlinIsland

KotlinIsland commented Apr 23, 2026

Copy link
Copy Markdown
Contributor Author

Callable[..., object] is right but not 100% sure.

not Callable, and not ..., as this is the gradual form. but (*object, **object), which is a non-denotable psuedo representation of the widest value for a parameter specification

@JelleZijlstra

Copy link
Copy Markdown
Member

I don't think that's right. The form we want here should be a supertype of every other possible value, and something like a one-parameter callable is not a subtype of (*: object, **: object).

@AlexWaygood

Copy link
Copy Markdown
Member

In ty we have a type that we spell as Top[Callable[..., object]] in ty's type display (a fully static type describing the infinite union of all possible Callable types). It's not representable using standard Python notation currently.

@KotlinIsland KotlinIsland requested a review from carljm April 28, 2026 05:21
charliermarsh added a commit to astral-sh/ruff that referenced this pull request Apr 29, 2026
## Summary

This PR adds `covariant`, `contravariant`, and `infer_variance` support
to `ParamSpec`.

See: python/typing#2215.

See:
#24479 (review).
@KotlinIsland

Copy link
Copy Markdown
Contributor Author

i've pushed changes, it's good to review again

Comment thread conformance/tests/generics_typevartuple_variance.py Outdated

@JelleZijlstra JelleZijlstra left a comment

Copy link
Copy Markdown
Member

Choose a reason for hiding this comment

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

Agree with this idea, we might have to tweak the exact PR a bit more.

thejchap pushed a commit to thejchap/ruff that referenced this pull request May 23, 2026
## Summary

This PR adds `covariant`, `contravariant`, and `infer_variance` support
to `ParamSpec`.

See: python/typing#2215.

See:
astral-sh#24479 (review).

@davidhalter davidhalter left a comment

Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

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

I like the general approach. I think it would also be useful to have at least one type checker with a proper implementation of this new feature.

Comment thread docs/spec/generics.rst Outdated

an ``object`` instance for a type variable.
a ``*tuple[object, ...]`` value for a type variable tuple.
a for a parameter specification, a 'top signature' value, i.e. a type

Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

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

I assume a for a is not wanted? The whole paragraph feels a bit weird and the formatting might be a bit off.

@KotlinIsland KotlinIsland Jun 22, 2026

Copy link
Copy Markdown
Contributor Author

Choose a reason for hiding this comment

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

addressed points 1 and 3. do you have a suggestion on how to improve the wording?

Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

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

I would probably go with a list here:

Suggested change
a for a parameter specification, a 'top signature' value, i.e. a type
- an ``object`` instance for a type variable
- a ``*tuple[object, ...]`` value for a type variable tuple
- a "top signature" value for a parameter specification, i.e. a type that represents the super type of every possible signature.

Copy link
Copy Markdown
Contributor Author

Choose a reason for hiding this comment

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

done

Comment thread conformance/tests/generics_typevartuple_basic.py Outdated
@KotlinIsland

KotlinIsland commented Jun 22, 2026

Copy link
Copy Markdown
Contributor Author

proper implementation of this new feature.

see the mr description, it's implemented in pycharm and basedpyright

@KotlinIsland KotlinIsland requested a review from davidhalter June 22, 2026 14:00
@KotlinIsland KotlinIsland force-pushed the paramspec-variance branch 2 times, most recently from fdb31bb to 94b1653 Compare June 22, 2026 14:02
out_obj: CovariantTypeVarTuple[object] = CovariantTypeVarTuple[int]() # OK
out_multiple: CovariantTypeVarTuple[float, float] = CovariantTypeVarTuple[
int, # OK
object, # E

Copy link
Copy Markdown
Collaborator

Choose a reason for hiding this comment

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

I don't think we should require that the error be reported on this exact line. Generally, the conformance tests allow an error to be reported on any reasonable line. Furthermore, I tested this snippet:

from typing import Generic, TypeVar

T1 = TypeVar("T1", covariant=True)
T2 = TypeVar("T2", covariant=True)

class C(Generic[T1, T2]): pass

x: C[float, float] = C[
    int,
    object,
]()

in mypy, pyright, pyrefly, and ty, and they all reported the error either on the first line of x's definition or on the entire definition, which presumably the current conformance tests accept, so it would be odd to have a different requirement for TypeVarTuple than for TypeVar.

Copy link
Copy Markdown
Contributor Author

Choose a reason for hiding this comment

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

okay, addressed with a matrix of four assignments

@KotlinIsland KotlinIsland requested a review from rchen152 June 23, 2026 02:00
@KotlinIsland KotlinIsland force-pushed the paramspec-variance branch 2 times, most recently from 997d9a6 to 7c7dc6a Compare June 23, 2026 02:03
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Labels

topic: typing spec For improving the typing spec

Projects

None yet

Development

Successfully merging this pull request may close these issues.

8 participants