-
-
Notifications
You must be signed in to change notification settings - Fork 2.9k
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
mypy not inferring that variable cannot be None after assignment from dict.get #4805
Comments
Good research! I can also make the error go away through a temporary variable: a = d.get("a", "foo bar")
arg = a I've got a feeling it's an bug in the type solver. |
In my project, I have code like: import typing
class Task:
def __init__(self, name: typing.Union[None, str] = None) -> None:
self.name = name
if not self.name:
self.name = "Komu"
def caps(self) -> str:
return self.name.upper() when I run mypy on it I get: I'm been forced to now right my code as: class Task:
def caps(self) -> str:
# this assert is here to make mypy happy
assert isinstance(self.name, str)
return self.name.upper() |
@komuw That looks like a different problem. Can you open a new issue for it? |
@komuw This is unrelated to this issue, and not actually a bug, mypy can't guess your intention for the ("long living" and externally visible) class Task:
def __init__(self, name: Union[None, str] = None) -> None:
if name is not None:
self.name = name
else:
self.name = "Komu" Or even simpler (assuming this is the actual code you wanted to type): class Task:
def __init__(self, name: str = "Komu") -> None:
self.name = name |
thanks for the pointer, re-writing as suggested makes mypy happy and is what I'll result to in my project. self.name = name
if not self.name:
self.name = "Komu" if name is not None:
self.name = name
else:
self.name = "Komu" |
What: - remove the asserts that had been added to make `mypy` happy Why: - After seeking clarification[1] on the `mypy` issue tracker, a workaround was found References: 1. python/mypy#4805
No, it shouldn't. Within the method body they result in same types for If we would infer task = Task()
name: Optional[str]
task.name = name # Should this be valid? Since this is a backwards incompatible change with high potential for breaking currently passing code, there is no way this will be changed. |
I'm having a similar issue, but with Callables. def outer(func: Optional[Callable] = None) -> Callable:
if func is None:
func = lambda: None
def inner(*args, **kwargs) -> Any:
return func(*args, **kwargs)
return inner
As gvanrossum mentioned, the workaround of assigning it to a new variable makes the error go away. def outer(
func: Optional[Callable] = None,
) -> Callable:
if func is None:
func = lambda: None
my_func = func
def inner(*args, **kwargs) -> Any:
'''Inner function'''
return my_func(*args, **kwargs)
return inner
|
Issue: python/mypy#4805 Signed-off-by: NaNAGISaSA <[email protected]>
I've come across this issue (I believe). Sharing a similar piece of code in case it helps in troubleshooting.
With mypy 0.910 |
Hello, any update on this? I've encountered a very similar problem with |
The same thing happens if you guard against Nones with an exception: def __init__(self, my_var_id: str, my_var: Optional[MyVar] = None):
self._my_var = my_var
if self._my_var is None:
self._my_var = self._lookup_my_var(my_var_id) # Function that may retrieve a result i.e. from a DB
if self._my_var is None:
raise Exception("My var does not exist")
|
In my case, I was certain that the instance is not # logger.parent is defined as Optional[logging.Logger]
prefect_logger = cast(logging.Logger, logger.parent) One can argue if this is elegant or not, but this is a solution I'm happy with (it silenced the FA). |
@shaybensasson this solution worked for me as well, when I knew that if I didn't return by a certain point my variable was guaranteed not to be None |
@shaybensasson, rather than using a |
Fixes #4805 Fixes #13936 It is known that mypy can overuse outer type context sometimes (especially when it is a union). This prevents a common use case for narrowing types (see issue and test cases). This is a somewhat major semantic change, but I think it should match better what a user would expect.
Code
Expected output
None, since the code is correctly typed (afaict, at least)
Actual output
with
--strict-optional
:Inserting
reveal_type(arg)
before the print givesUnion[builtins.str, builtins.None]
.The following typechecks fine, so mypy definitely understands the return type of
dict.get
with two arguments:Similarly, mypy can infer that
arg
cannot be None after anis None
and simple assignment:but combining the two fails.
The text was updated successfully, but these errors were encountered: