Skip to content

Commit d2cf9c6

Browse files
authored
New semantic analyzer: fix type promotions special cases (#6609)
Set up type promotions after the main semantic analyzer pass to avoid the need to defer. This fixes an issue with test stubs, but the existing approach probably worked with typeshed. Another benefit of this approach is that things are easier to reason about, as the main semantic analysis pass is a bit simpler. This fixes the testDivmod type checker test case.
1 parent 1f10981 commit d2cf9c6

File tree

5 files changed

+70
-55
lines changed

5 files changed

+70
-55
lines changed

mypy/checkexpr.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -2260,7 +2260,7 @@ def check_op(self, method: str, base_type: Type,
22602260

22612261
# Step 1: We first try leaving the right arguments alone and destructure
22622262
# just the left ones. (Mypy can sometimes perform some more precise inference
2263-
# if we leave the right operands a union -- see testOperatorWithEmptyListAndSum.
2263+
# if we leave the right operands a union -- see testOperatorWithEmptyListAndSum.)
22642264
msg = self.msg.clean_copy()
22652265
msg.disable_count = 0
22662266
all_results = []

mypy/newsemanal/semanal.py

-50
Original file line numberDiff line numberDiff line change
@@ -127,36 +127,6 @@
127127
'typing.typevar': 'typing.TypeVar',
128128
} # type: Final
129129

130-
# Hard coded type promotions (shared between all Python versions).
131-
# These add extra ad-hoc edges to the subtyping relation. For example,
132-
# int is considered a subtype of float, even though there is no
133-
# subclass relationship.
134-
TYPE_PROMOTIONS = {
135-
'builtins.int': 'builtins.float',
136-
'builtins.float': 'builtins.complex',
137-
} # type: Final
138-
139-
# Hard coded type promotions for Python 3.
140-
#
141-
# Note that the bytearray -> bytes promotion is a little unsafe
142-
# as some functions only accept bytes objects. Here convenience
143-
# trumps safety.
144-
TYPE_PROMOTIONS_PYTHON3 = TYPE_PROMOTIONS.copy() # type: Final
145-
TYPE_PROMOTIONS_PYTHON3.update({
146-
'builtins.bytearray': 'builtins.bytes',
147-
})
148-
149-
# Hard coded type promotions for Python 2.
150-
#
151-
# These promotions are unsafe, but we are doing them anyway
152-
# for convenience and also for Python 3 compatibility
153-
# (bytearray -> str).
154-
TYPE_PROMOTIONS_PYTHON2 = TYPE_PROMOTIONS.copy() # type: Final
155-
TYPE_PROMOTIONS_PYTHON2.update({
156-
'builtins.str': 'builtins.unicode',
157-
'builtins.bytearray': 'builtins.str',
158-
})
159-
160130
# Map from the full name of a missing definition to the test fixture (under
161131
# test-data/unit/fixtures/) that provides the definition. This is used for
162132
# generating better error messages when running mypy tests only.
@@ -918,7 +888,6 @@ def analyze_class(self, defn: ClassDef) -> None:
918888
for decorator in defn.decorators:
919889
self.analyze_class_decorator(defn, decorator)
920890
self.analyze_class_body_common(defn)
921-
self.setup_type_promotion(defn)
922891

923892
def is_core_builtin_class(self, defn: ClassDef) -> bool:
924893
return self.cur_mod_id == 'builtins' and defn.name in CORE_BUILTIN_CLASSES
@@ -1017,25 +986,6 @@ def analyze_class_decorator(self, defn: ClassDef, decorator: Expression) -> None
1017986
'typing_extensions.final'):
1018987
defn.info.is_final = True
1019988

1020-
def setup_type_promotion(self, defn: ClassDef) -> None:
1021-
"""Setup extra, ad-hoc subtyping relationships between classes (promotion).
1022-
1023-
This includes things like 'int' being compatible with 'float'.
1024-
"""
1025-
promote_target = None # type: Optional[Type]
1026-
for decorator in defn.decorators:
1027-
if isinstance(decorator, CallExpr):
1028-
analyzed = decorator.analyzed
1029-
if isinstance(analyzed, PromoteExpr):
1030-
# _promote class decorator (undocumented feature).
1031-
promote_target = analyzed.type
1032-
if not promote_target:
1033-
promotions = (TYPE_PROMOTIONS_PYTHON3 if self.options.python_version[0] >= 3
1034-
else TYPE_PROMOTIONS_PYTHON2)
1035-
if defn.fullname in promotions:
1036-
promote_target = self.named_type_or_none(promotions[defn.fullname])
1037-
defn.info._promote = promote_target
1038-
1039989
def clean_up_bases_and_infer_type_variables(
1040990
self,
1041991
defn: ClassDef,

mypy/newsemanal/semanal_classprop.py

+66-2
Original file line numberDiff line numberDiff line change
@@ -5,9 +5,47 @@
55

66
from typing import List, Set, Optional
77

8-
from mypy.nodes import Node, TypeInfo, Var, Decorator, OverloadedFuncDef
9-
from mypy.types import Instance
8+
from mypy.nodes import (
9+
Node, TypeInfo, Var, Decorator, OverloadedFuncDef, SymbolTable, CallExpr, PromoteExpr,
10+
)
11+
from mypy.types import Instance, Type
1012
from mypy.errors import Errors
13+
from mypy.options import Options
14+
15+
MYPY = False
16+
if MYPY:
17+
from typing_extensions import Final
18+
19+
20+
# Hard coded type promotions (shared between all Python versions).
21+
# These add extra ad-hoc edges to the subtyping relation. For example,
22+
# int is considered a subtype of float, even though there is no
23+
# subclass relationship.
24+
TYPE_PROMOTIONS = {
25+
'builtins.int': 'float',
26+
'builtins.float': 'complex',
27+
} # type: Final
28+
29+
# Hard coded type promotions for Python 3.
30+
#
31+
# Note that the bytearray -> bytes promotion is a little unsafe
32+
# as some functions only accept bytes objects. Here convenience
33+
# trumps safety.
34+
TYPE_PROMOTIONS_PYTHON3 = TYPE_PROMOTIONS.copy() # type: Final
35+
TYPE_PROMOTIONS_PYTHON3.update({
36+
'builtins.bytearray': 'bytes',
37+
})
38+
39+
# Hard coded type promotions for Python 2.
40+
#
41+
# These promotions are unsafe, but we are doing them anyway
42+
# for convenience and also for Python 3 compatibility
43+
# (bytearray -> str).
44+
TYPE_PROMOTIONS_PYTHON2 = TYPE_PROMOTIONS.copy() # type: Final
45+
TYPE_PROMOTIONS_PYTHON2.update({
46+
'builtins.str': 'unicode',
47+
'builtins.bytearray': 'str',
48+
})
1149

1250

1351
def calculate_class_abstract_status(typ: TypeInfo, is_stub_file: bool, errors: Errors) -> None:
@@ -94,3 +132,29 @@ def calculate_class_vars(info: TypeInfo) -> None:
94132
and isinstance(member.node, Var)
95133
and member.node.is_classvar):
96134
node.is_classvar = True
135+
136+
137+
def add_type_promotion(info: TypeInfo, module_names: SymbolTable, options: Options) -> None:
138+
"""Setup extra, ad-hoc subtyping relationships between classes (promotion).
139+
140+
This includes things like 'int' being compatible with 'float'.
141+
"""
142+
defn = info.defn
143+
promote_target = None # type: Optional[Type]
144+
for decorator in defn.decorators:
145+
if isinstance(decorator, CallExpr):
146+
analyzed = decorator.analyzed
147+
if isinstance(analyzed, PromoteExpr):
148+
# _promote class decorator (undocumented feature).
149+
promote_target = analyzed.type
150+
if not promote_target:
151+
promotions = (TYPE_PROMOTIONS_PYTHON3 if options.python_version[0] >= 3
152+
else TYPE_PROMOTIONS_PYTHON2)
153+
if defn.fullname in promotions:
154+
target_sym = module_names.get(promotions[defn.fullname])
155+
# With test stubs, the target may not exist.
156+
if target_sym:
157+
target_info = target_sym.node
158+
assert isinstance(target_info, TypeInfo)
159+
promote_target = Instance(target_info, [])
160+
defn.info._promote = promote_target

mypy/newsemanal/semanal_main.py

+3-1
Original file line numberDiff line numberDiff line change
@@ -36,7 +36,8 @@
3636
NewSemanticAnalyzer, apply_semantic_analyzer_patches, remove_imported_names_from_symtable
3737
)
3838
from mypy.newsemanal.semanal_classprop import (
39-
calculate_class_abstract_status, calculate_class_vars, check_protocol_status
39+
calculate_class_abstract_status, calculate_class_vars, check_protocol_status,
40+
add_type_promotion
4041
)
4142
from mypy.errors import Errors
4243
from mypy.newsemanal.semanal_infer import infer_decorator_signature_if_simple
@@ -327,6 +328,7 @@ def calculate_class_properties(graph: 'Graph', scc: List[str], errors: Errors) -
327328
calculate_class_abstract_status(node.node, tree.is_stub, errors)
328329
check_protocol_status(node.node, errors)
329330
calculate_class_vars(node.node)
331+
add_type_promotion(node.node, tree.names, graph[module].options)
330332

331333

332334
def check_blockers(graph: 'Graph', scc: List[str]) -> None:

mypy/test/hacks.py

-1
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,6 @@
88
# Files to not run with new semantic analyzer.
99
new_semanal_blacklist = [
1010
'check-async-await.test',
11-
'check-expressions.test',
1211
'check-flags.test',
1312
'check-incremental.test',
1413
'check-literal.test',

0 commit comments

Comments
 (0)