Skip to content

Commit 5b488ab

Browse files
ilevkivskyiJukkaL
authored andcommitted
Optimize Unpack for failures (#15967)
This is a small but possibly important PR. Wherever possible we should represent user error and/or failed type inference as `*tuple[Any, ...]`/`*tuple[<nothing>, ...]`, rather than `Unpack[Any]`/`Unpack[<nothing>]` or plain `Any`/`<nothing>`. This way we will not need any special casing for failure conditions in various places without risking a crash instead of a graceful failure (error message).
1 parent 4c963c9 commit 5b488ab

File tree

4 files changed

+24
-27
lines changed

4 files changed

+24
-27
lines changed

mypy/expandtype.py

+6-17
Original file line numberDiff line numberDiff line change
@@ -273,7 +273,7 @@ def visit_unpack_type(self, t: UnpackType) -> Type:
273273
# example is non-normalized types when called from semanal.py.
274274
return UnpackType(t.type.accept(self))
275275

276-
def expand_unpack(self, t: UnpackType) -> list[Type] | AnyType | UninhabitedType:
276+
def expand_unpack(self, t: UnpackType) -> list[Type]:
277277
assert isinstance(t.type, TypeVarTupleType)
278278
repl = get_proper_type(self.variables.get(t.type.id, t.type))
279279
if isinstance(repl, TupleType):
@@ -285,9 +285,9 @@ def expand_unpack(self, t: UnpackType) -> list[Type] | AnyType | UninhabitedType
285285
):
286286
return [UnpackType(typ=repl)]
287287
elif isinstance(repl, (AnyType, UninhabitedType)):
288-
# tuple[Any, ...] for Any would be better, but we don't have
289-
# the type info to construct that type here.
290-
return repl
288+
# Replace *Ts = Any with *Ts = *tuple[Any, ...] and some for <nothing>.
289+
# These types may appear here as a result of user error or failed inference.
290+
return [UnpackType(t.type.tuple_fallback.copy_modified(args=[repl]))]
291291
else:
292292
raise RuntimeError(f"Invalid type replacement to expand: {repl}")
293293

@@ -310,12 +310,7 @@ def interpolate_args_for_unpack(self, t: CallableType, var_arg: UnpackType) -> l
310310
# We have plain Unpack[Ts]
311311
assert isinstance(var_arg_type, TypeVarTupleType)
312312
fallback = var_arg_type.tuple_fallback
313-
expanded_items_res = self.expand_unpack(var_arg)
314-
if isinstance(expanded_items_res, list):
315-
expanded_items = expanded_items_res
316-
else:
317-
# We got Any or <nothing>
318-
return prefix + [expanded_items_res] + suffix
313+
expanded_items = self.expand_unpack(var_arg)
319314
new_unpack = UnpackType(TupleType(expanded_items, fallback))
320315
return prefix + [new_unpack] + suffix
321316

@@ -394,14 +389,8 @@ def expand_types_with_unpack(
394389
items: list[Type] = []
395390
for item in typs:
396391
if isinstance(item, UnpackType) and isinstance(item.type, TypeVarTupleType):
397-
unpacked_items = self.expand_unpack(item)
398-
if isinstance(unpacked_items, (AnyType, UninhabitedType)):
399-
# TODO: better error for <nothing>, something like tuple of unknown?
400-
return unpacked_items
401-
else:
402-
items.extend(unpacked_items)
392+
items.extend(self.expand_unpack(item))
403393
else:
404-
# Must preserve original aliases when possible.
405394
items.append(item.accept(self))
406395
return items
407396

mypy/semanal_main.py

+2
Original file line numberDiff line numberDiff line change
@@ -381,6 +381,7 @@ def check_type_arguments(graph: Graph, scc: list[str], errors: Errors) -> None:
381381
errors,
382382
state.options,
383383
is_typeshed_file(state.options.abs_custom_typeshed_dir, state.path or ""),
384+
state.manager.semantic_analyzer.named_type,
384385
)
385386
with state.wrap_context():
386387
with mypy.state.state.strict_optional_set(state.options.strict_optional):
@@ -399,6 +400,7 @@ def check_type_arguments_in_targets(
399400
errors,
400401
state.options,
401402
is_typeshed_file(state.options.abs_custom_typeshed_dir, state.path or ""),
403+
state.manager.semantic_analyzer.named_type,
402404
)
403405
with state.wrap_context():
404406
with mypy.state.state.strict_optional_set(state.options.strict_optional):

mypy/semanal_typeargs.py

+14-7
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@
77

88
from __future__ import annotations
99

10-
from typing import Sequence
10+
from typing import Callable, Sequence
1111

1212
from mypy import errorcodes as codes, message_registry
1313
from mypy.errorcodes import ErrorCode
@@ -42,11 +42,18 @@
4242

4343

4444
class TypeArgumentAnalyzer(MixedTraverserVisitor):
45-
def __init__(self, errors: Errors, options: Options, is_typeshed_file: bool) -> None:
45+
def __init__(
46+
self,
47+
errors: Errors,
48+
options: Options,
49+
is_typeshed_file: bool,
50+
named_type: Callable[[str, list[Type]], Instance],
51+
) -> None:
4652
super().__init__()
4753
self.errors = errors
4854
self.options = options
4955
self.is_typeshed_file = is_typeshed_file
56+
self.named_type = named_type
5057
self.scope = Scope()
5158
# Should we also analyze function definitions, or only module top-levels?
5259
self.recurse_into_functions = True
@@ -243,16 +250,16 @@ def visit_unpack_type(self, typ: UnpackType) -> None:
243250
return
244251
if isinstance(proper_type, TypeVarTupleType):
245252
return
253+
# TODO: this should probably be .has_base("builtins.tuple"), also elsewhere.
246254
if isinstance(proper_type, Instance) and proper_type.type.fullname == "builtins.tuple":
247255
return
248-
if isinstance(proper_type, AnyType) and proper_type.type_of_any == TypeOfAny.from_error:
249-
return
250-
if not isinstance(proper_type, UnboundType):
251-
# Avoid extra errors if there were some errors already.
256+
if not isinstance(proper_type, (UnboundType, AnyType)):
257+
# Avoid extra errors if there were some errors already. Also interpret plain Any
258+
# as tuple[Any, ...] (this is better for the code in type checker).
252259
self.fail(
253260
message_registry.INVALID_UNPACK.format(format_type(proper_type, self.options)), typ
254261
)
255-
typ.type = AnyType(TypeOfAny.from_error)
262+
typ.type = self.named_type("builtins.tuple", [AnyType(TypeOfAny.from_error)])
256263

257264
def check_type_var_values(
258265
self, name: str, actuals: list[Type], arg_name: str, valids: list[Type], context: Context

test-data/unit/check-typevar-tuple.test

+2-3
Original file line numberDiff line numberDiff line change
@@ -17,16 +17,15 @@ reveal_type(f(args)) # N: Revealed type is "Tuple[builtins.int, builtins.str]"
1717

1818
reveal_type(f(varargs)) # N: Revealed type is "builtins.tuple[builtins.int, ...]"
1919

20-
if object():
21-
f(0) # E: Argument 1 to "f" has incompatible type "int"; expected <nothing>
20+
f(0) # E: Argument 1 to "f" has incompatible type "int"; expected "Tuple[<nothing>, ...]"
2221

2322
def g(a: Tuple[Unpack[Ts]], b: Tuple[Unpack[Ts]]) -> Tuple[Unpack[Ts]]:
2423
return a
2524

2625
reveal_type(g(args, args)) # N: Revealed type is "Tuple[builtins.int, builtins.str]"
2726
reveal_type(g(args, args2)) # N: Revealed type is "Tuple[builtins.int, builtins.str]"
2827
reveal_type(g(args, args3)) # N: Revealed type is "builtins.tuple[builtins.object, ...]"
29-
reveal_type(g(any, any)) # N: Revealed type is "Any"
28+
reveal_type(g(any, any)) # N: Revealed type is "builtins.tuple[Any, ...]"
3029
[builtins fixtures/tuple.pyi]
3130

3231
[case testTypeVarTupleMixed]

0 commit comments

Comments
 (0)