This question is to clarify my doubts related to python typing
from typing import Union
class ParentClass:
parent_prop = 1
class ChildA(ParentClass):
child_a_prop = 2
class ChildB(ParentClass):
child_b_prop = 3
def method_body(val) -> ParentClass:
if val:
return ChildA()
else:
return ChildB()
def another_method() -> ChildA:
return method_body(True)
print(another_method().child_a_prop)
In the above piece of code, the linting tool I used is printing error as below
error: Incompatible return value type (got "ParentClass", expected "ChildA")
(where I do method_body(True))
I have also set the method_body return type as Union[ChildA, ChildB].
This will result error: Incompatible return value type (got "Union[ChildA, ChildB]", expected "ChildA")
I am looking for a better way to do this. If anyone knows the solution, your help will be very much appreciated.
mypy does not do runtime analysis so it cannot guess that calling
method_body with argument True will always result in a ChildA object. So the error it produces does make sense.
You have to guide mypy in some way to tell him that you know what you are doing and that another_method indeed produces a ChildA object when called with argument True. One is to use a cast:
from typing import cast
def another_method() -> ChildA:
return cast(ChildA, method_body(True))
Another one is to add an assertion:
def another_method() -> ChildA:
result = method_body(True)
assert isinstance(result, ChildA)
return result
The difference between the two is that the cast does not have any runtime implication. You can think of it as a comment put here to guide mypy in its checks, but the cast function only returns its second parameter, ie, here is the body of cast:
def cast(typ, val):
return val
Whereas the assert can naturally raise an AssertionError error (not in that case obviously, but in general).
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With