KEMBAR78
Fix isinstance with type aliases to PEP 604 unions by hauntsaninja · Pull Request #17371 · python/mypy · GitHub
Skip to content

Conversation

@hauntsaninja
Copy link
Collaborator

@hauntsaninja hauntsaninja commented Jun 12, 2024

Fixes #12155, fixes #11673, seems pretty commonly reported issue

@github-actions

This comment has been minimized.

@github-actions

This comment has been minimized.

@github-actions

This comment has been minimized.

@github-actions
Copy link
Contributor

According to mypy_primer, this change doesn't affect type check results on a corpus of open source code. ✅

@hauntsaninja hauntsaninja changed the title Fix isinstance with PEP 604 type aliases Fix isinstance with type aliases to PEP 604 unions Jun 12, 2024
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.

What about old-style unions? They work at runtime:

>>> from typing import Union
>>> isinstance(1, Union[int, str])
True

It's also okay to only support PEP 604 unions for now.

@JelleZijlstra
Copy link
Member

Note those only work with isinstance() in 3.10+.

@hauntsaninja
Copy link
Collaborator Author

hauntsaninja commented Jun 13, 2024

I'll take a look at supporting old-style unions, but in another PR. It doesn't seem to be nearly as popular a feature request and it would require monkeypatching typeshed or doing more isinstance special casing: python/typeshed#12137

@hauntsaninja hauntsaninja merged commit 98a22c4 into python:master Jun 13, 2024
@hauntsaninja hauntsaninja deleted the fix-type-alias branch June 13, 2024 21:17
@hauntsaninja
Copy link
Collaborator Author

This also fixes #13810

@ngnpope
Copy link
Contributor

ngnpope commented Jul 22, 2024

@hauntsaninja Unfortunately it seems this fix doesn't extend to the case where the type alias is defined in another module:

Given a.py:

from typing import TypeAlias

IntOrStr: TypeAlias = int | str

assert isinstance(1, int | str)
assert isinstance(1, IntOrStr)

And b.py:

from a import IntOrStr

assert isinstance(1, int | str)
assert isinstance(1, IntOrStr)

I get the following output:

$ mypy --version
mypy 1.11.0 (compiled: yes)
$ mypy --strict a.py
Success: no issues found in 1 source file
$ mypy --strict b.py
b.py:4: error: Parameterized generics cannot be used with class or instance checks  [misc]
b.py:4: error: Argument 2 to "isinstance" has incompatible type "<typing special form>"; expected "_ClassInfo"  [arg-type]
Found 2 errors in 1 file (checked 1 source file)

The issue seems to be that uses_pep604_syntax is False on this line.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Labels

Projects

None yet

4 participants