12
12
import mypy .checkexpr
13
13
from mypy import errorcodes as codes , join , message_registry , nodes , operators
14
14
from mypy .binder import ConditionalTypeBinder , Frame , get_declaration
15
- from mypy .checkmember import (
16
- MemberContext ,
17
- analyze_decorator_or_funcbase_access ,
18
- analyze_descriptor_access ,
19
- analyze_member_access ,
20
- )
15
+ from mypy .checkmember import analyze_member_access
21
16
from mypy .checkpattern import PatternChecker
22
17
from mypy .constraints import SUPERTYPE_OF
23
18
from mypy .erasetype import erase_type , erase_typevars , remove_instance_last_known_values
@@ -3233,7 +3228,7 @@ def check_assignment(
3233
3228
)
3234
3229
else :
3235
3230
self .try_infer_partial_generic_type_from_assignment (lvalue , rvalue , "=" )
3236
- lvalue_type , index_lvalue , inferred = self .check_lvalue (lvalue )
3231
+ lvalue_type , index_lvalue , inferred = self .check_lvalue (lvalue , rvalue )
3237
3232
# If we're assigning to __getattr__ or similar methods, check that the signature is
3238
3233
# valid.
3239
3234
if isinstance (lvalue , NameExpr ) and lvalue .node :
@@ -4339,7 +4334,9 @@ def check_multi_assignment_from_iterable(
4339
4334
else :
4340
4335
self .msg .type_not_iterable (rvalue_type , context )
4341
4336
4342
- def check_lvalue (self , lvalue : Lvalue ) -> tuple [Type | None , IndexExpr | None , Var | None ]:
4337
+ def check_lvalue (
4338
+ self , lvalue : Lvalue , rvalue : Expression | None = None
4339
+ ) -> tuple [Type | None , IndexExpr | None , Var | None ]:
4343
4340
lvalue_type = None
4344
4341
index_lvalue = None
4345
4342
inferred = None
@@ -4357,7 +4354,7 @@ def check_lvalue(self, lvalue: Lvalue) -> tuple[Type | None, IndexExpr | None, V
4357
4354
elif isinstance (lvalue , IndexExpr ):
4358
4355
index_lvalue = lvalue
4359
4356
elif isinstance (lvalue , MemberExpr ):
4360
- lvalue_type = self .expr_checker .analyze_ordinary_member_access (lvalue , True )
4357
+ lvalue_type = self .expr_checker .analyze_ordinary_member_access (lvalue , True , rvalue )
4361
4358
self .store_type (lvalue , lvalue_type )
4362
4359
elif isinstance (lvalue , NameExpr ):
4363
4360
lvalue_type = self .expr_checker .analyze_ref_expr (lvalue , lvalue = True )
@@ -4704,12 +4701,8 @@ def check_member_assignment(
4704
4701
4705
4702
Return the inferred rvalue_type, inferred lvalue_type, and whether to use the binder
4706
4703
for this assignment.
4707
-
4708
- Note: this method exists here and not in checkmember.py, because we need to take
4709
- care about interaction between binder and __set__().
4710
4704
"""
4711
4705
instance_type = get_proper_type (instance_type )
4712
- attribute_type = get_proper_type (attribute_type )
4713
4706
# Descriptors don't participate in class-attribute access
4714
4707
if (isinstance (instance_type , FunctionLike ) and instance_type .is_type_obj ()) or isinstance (
4715
4708
instance_type , TypeType
@@ -4721,107 +4714,16 @@ def check_member_assignment(
4721
4714
get_lvalue_type = self .expr_checker .analyze_ordinary_member_access (
4722
4715
lvalue , is_lvalue = False
4723
4716
)
4724
- use_binder = is_same_type (get_lvalue_type , attribute_type )
4725
-
4726
- if not isinstance (attribute_type , Instance ):
4727
- # TODO: support __set__() for union types.
4728
- rvalue_type , _ = self .check_simple_assignment (attribute_type , rvalue , context )
4729
- return rvalue_type , attribute_type , use_binder
4730
-
4731
- mx = MemberContext (
4732
- is_lvalue = False ,
4733
- is_super = False ,
4734
- is_operator = False ,
4735
- original_type = instance_type ,
4736
- context = context ,
4737
- self_type = None ,
4738
- chk = self ,
4739
- )
4740
- get_type = analyze_descriptor_access (attribute_type , mx , assignment = True )
4741
- if not attribute_type .type .has_readable_member ("__set__" ):
4742
- # If there is no __set__, we type-check that the assigned value matches
4743
- # the return type of __get__. This doesn't match the python semantics,
4744
- # (which allow you to override the descriptor with any value), but preserves
4745
- # the type of accessing the attribute (even after the override).
4746
- rvalue_type , _ = self .check_simple_assignment (get_type , rvalue , context )
4747
- return rvalue_type , get_type , use_binder
4748
-
4749
- dunder_set = attribute_type .type .get_method ("__set__" )
4750
- if dunder_set is None :
4751
- self .fail (
4752
- message_registry .DESCRIPTOR_SET_NOT_CALLABLE .format (
4753
- attribute_type .str_with_options (self .options )
4754
- ),
4755
- context ,
4756
- )
4757
- return AnyType (TypeOfAny .from_error ), get_type , False
4758
-
4759
- bound_method = analyze_decorator_or_funcbase_access (
4760
- defn = dunder_set ,
4761
- itype = attribute_type ,
4762
- name = "__set__" ,
4763
- mx = mx .copy_modified (self_type = attribute_type ),
4764
- )
4765
- typ = map_instance_to_supertype (attribute_type , dunder_set .info )
4766
- dunder_set_type = expand_type_by_instance (bound_method , typ )
4767
-
4768
- callable_name = self .expr_checker .method_fullname (attribute_type , "__set__" )
4769
- dunder_set_type = self .expr_checker .transform_callee_type (
4770
- callable_name ,
4771
- dunder_set_type ,
4772
- [TempNode (instance_type , context = context ), rvalue ],
4773
- [nodes .ARG_POS , nodes .ARG_POS ],
4774
- context ,
4775
- object_type = attribute_type ,
4776
- )
4777
-
4778
- # For non-overloaded setters, the result should be type-checked like a regular assignment.
4779
- # Hence, we first only try to infer the type by using the rvalue as type context.
4780
- type_context = rvalue
4781
- with self .msg .filter_errors ():
4782
- _ , inferred_dunder_set_type = self .expr_checker .check_call (
4783
- dunder_set_type ,
4784
- [TempNode (instance_type , context = context ), type_context ],
4785
- [nodes .ARG_POS , nodes .ARG_POS ],
4786
- context ,
4787
- object_type = attribute_type ,
4788
- callable_name = callable_name ,
4789
- )
4790
-
4791
- # And now we in fact type check the call, to show errors related to wrong arguments
4792
- # count, etc., replacing the type context for non-overloaded setters only.
4793
- inferred_dunder_set_type = get_proper_type (inferred_dunder_set_type )
4794
- if isinstance (inferred_dunder_set_type , CallableType ):
4795
- type_context = TempNode (AnyType (TypeOfAny .special_form ), context = context )
4796
- self .expr_checker .check_call (
4797
- dunder_set_type ,
4798
- [TempNode (instance_type , context = context ), type_context ],
4799
- [nodes .ARG_POS , nodes .ARG_POS ],
4800
- context ,
4801
- object_type = attribute_type ,
4802
- callable_name = callable_name ,
4803
- )
4804
-
4805
- # Search for possible deprecations:
4806
- mx .chk .check_deprecated (dunder_set , mx .context )
4807
- mx .chk .warn_deprecated_overload_item (
4808
- dunder_set , mx .context , target = inferred_dunder_set_type , selftype = attribute_type
4809
- )
4810
4717
4811
- # In the following cases, a message already will have been recorded in check_call.
4812
- if (not isinstance (inferred_dunder_set_type , CallableType )) or (
4813
- len (inferred_dunder_set_type .arg_types ) < 2
4814
- ):
4815
- return AnyType (TypeOfAny .from_error ), get_type , False
4816
-
4817
- set_type = inferred_dunder_set_type .arg_types [1 ]
4818
4718
# Special case: if the rvalue_type is a subtype of both '__get__' and '__set__' types,
4819
4719
# and '__get__' type is narrower than '__set__', then we invoke the binder to narrow type
4820
4720
# by this assignment. Technically, this is not safe, but in practice this is
4821
4721
# what a user expects.
4822
- rvalue_type , _ = self .check_simple_assignment (set_type , rvalue , context )
4823
- infer = is_subtype (rvalue_type , get_type ) and is_subtype (get_type , set_type )
4824
- return rvalue_type if infer else set_type , get_type , infer
4722
+ rvalue_type , _ = self .check_simple_assignment (attribute_type , rvalue , context )
4723
+ infer = is_subtype (rvalue_type , get_lvalue_type ) and is_subtype (
4724
+ get_lvalue_type , attribute_type
4725
+ )
4726
+ return rvalue_type if infer else attribute_type , attribute_type , infer
4825
4727
4826
4728
def check_indexed_assignment (
4827
4729
self , lvalue : IndexExpr , rvalue : Expression , context : Context
0 commit comments