-
-
Notifications
You must be signed in to change notification settings - Fork 2.8k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Protocols and ... if ... else ...
#5392
Comments
On further thought, I wonder why mypy tries to infer the base type for class Foo:
def x(self) -> None: ...
class Bar:
def x(self) -> None: ...
o = Foo() if bool() else Bar()
o.x() |
Continuing my train of thoughts: Changing class Base: ...
class Foo(Base): ...
class Bar(Base): ...
class Baz(Base): ...
x = Foo() if bool() else Bar()
...
x = Baz() Although I'd argue that in this case an explicit annotation for Strangely enough, the following code does type check: from typing import Union
def foo(x: Union[int, str]) -> None: ...
foo("" if bool() else 0) while the following does not: "" if bool() else 0
foo(x) |
Thanks for opening this issue. There's been discussion about this. There are pros and cons for returning unions for conditional expressions. It's my day off and I don't recall where the larger discussion lives. :-( |
We looked at changing if/else expressions to always have union types, but it caused a lot of issues with code that relied on the existing semantics. Instead, we only modified it to infer a union type if the type context is a union type. Similarly, we could infer a union/protocol type when the type context is a protocol type. Example: from typing_extensions import Protocol
class P(Protocol):
x: int
class A:
x: int
class B:
x: int
def f(p: P) -> None: pass
f(A() if bool() else B()) # Should probably be okay
y: P = A() if bool() else B() # Should probably be okay This change wouldn't directly fix the original issue, but it would at least provide a somewhat simple workaround: from typing import SupportsFloat
class Floaty:
def __float__(self): ...
x: SupportsFloat = reveal_type(Floaty() if bool() else 0) # Would be SupportsFloat This would clearly still be a compromise, but marginally better than what we currently have. It would also be consistent with how type inference issues are generally resolved -- by adding explicit type annotations. The fundamental problem is that sometimes there are multiple reasonable types we could infer, and we usually try to pick one of the options that we expect to be usually (but not always) a good one. |
This compromise seems reasonable, although I'd argue that a "correct" solution would be better. I understand that that means changes to existing code bases, but I feel that typing and mypy are still in semi-experimental states (the This particular case irks me not only because of the practical problem discussed in the referenced issue, but also because of the inconsistency with from typing import Protocol
class P(Protocol):
x: int
class Foo:
x: int
class Bar:
x: int
reveal_type(Foo() or Bar()) # Union[foo.Foo, foo.Bar]
reveal_type(Foo() if bool() else Bar()) # builtins.object |
I agree that this is something that would be nice to fix, but the core team doesn't have the bandwidth to deal with the fallout right now :-( The mypy fix is actually trivial, but fixing our internal Dropbox codebases would be a lot of work. The situation would likely be similar for many other mypy users with large codebases. Another alternative would be to allow the old behavior to be retained with a configuration option. |
I encountered this issue and made a test case when figuring out whether my code was incorrectly hinted or mypy was rejecting it unnecessarily. Perhaps the test case will be useful in implementing or discussing the handling of conditional expressions. from typing_extensions import Protocol
class Unary(Protocol):
def process(self, n: int) -> int:
pass
class Square:
def process(self, n: int) -> int:
return n ** 2
class Negate:
def process(self, n: int) -> int:
return -n
def f(cond: bool) -> Unary:
return Square() if cond else Negate()
def g(cond: bool) -> Unary:
if cond:
return Square()
else:
return Negate() What is interesting about this test case, is that for |
I would be open to work on this issue if someone can give me some pointers 😊 |
Fixed by #17427 |
python/typeshed#2323 exposed a problem with mypy's (lack of) handling of protocols in
... if ... else ...
constructs. Please consider:This will reveal the type of the first expression to be
builtins.object
, while the latter istyping.SupportsFloat
. This means that mypy (tested with 0.620, Python 3.7.0) will reject, for examplefloat(Floaty() if bool() else 0)
.The text was updated successfully, but these errors were encountered: