|
5 | 5 |
|
6 | 6 | from typing import List, Set, Optional
|
7 | 7 |
|
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 |
10 | 12 | 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 | +}) |
11 | 49 |
|
12 | 50 |
|
13 | 51 | 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:
|
94 | 132 | and isinstance(member.node, Var)
|
95 | 133 | and member.node.is_classvar):
|
96 | 134 | 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 |
0 commit comments