mirror of
				https://github.com/django/django.git
				synced 2025-10-25 14:46:09 +00:00 
			
		
		
		
	Fixed #31750 -- Made models.Field equality compare models for inherited fields.
This commit is contained in:
		
				
					committed by
					
						 Mariusz Felisiak
						Mariusz Felisiak
					
				
			
			
				
	
			
			
			
						parent
						
							453967477e
						
					
				
				
					commit
					502e75f9ed
				
			| @@ -516,17 +516,37 @@ class Field(RegisterLookupMixin): | ||||
|     def __eq__(self, other): | ||||
|         # Needed for @total_ordering | ||||
|         if isinstance(other, Field): | ||||
|             return self.creation_counter == other.creation_counter | ||||
|             return ( | ||||
|                 self.creation_counter == other.creation_counter and | ||||
|                 getattr(self, 'model', None) == getattr(other, 'model', None) | ||||
|             ) | ||||
|         return NotImplemented | ||||
|  | ||||
|     def __lt__(self, other): | ||||
|         # This is needed because bisect does not take a comparison function. | ||||
|         # Order by creation_counter first for backward compatibility. | ||||
|         if isinstance(other, Field): | ||||
|             if ( | ||||
|                 self.creation_counter != other.creation_counter or | ||||
|                 not hasattr(self, 'model') and not hasattr(other, 'model') | ||||
|             ): | ||||
|                 return self.creation_counter < other.creation_counter | ||||
|             elif hasattr(self, 'model') != hasattr(other, 'model'): | ||||
|                 return not hasattr(self, 'model')  # Order no-model fields first | ||||
|             else: | ||||
|                 # creation_counter's are equal, compare only models. | ||||
|                 return ( | ||||
|                     (self.model._meta.app_label, self.model._meta.model_name) < | ||||
|                     (other.model._meta.app_label, other.model._meta.model_name) | ||||
|                 ) | ||||
|         return NotImplemented | ||||
|  | ||||
|     def __hash__(self): | ||||
|         return hash(self.creation_counter) | ||||
|         return hash(( | ||||
|             self.creation_counter, | ||||
|             self.model._meta.app_label if hasattr(self, 'model') else None, | ||||
|             self.model._meta.model_name if hasattr(self, 'model') else None, | ||||
|         )) | ||||
|  | ||||
|     def __deepcopy__(self, memodict): | ||||
|         # We don't have to deepcopy very much here, since most things are not | ||||
|   | ||||
| @@ -494,6 +494,10 @@ Miscellaneous | ||||
|   instead of :exc:`~django.core.exceptions.SuspiciousOperation` when a session | ||||
|   is destroyed in a concurrent request. | ||||
|  | ||||
| * The :class:`django.db.models.Field` equality operator now correctly | ||||
|   distinguishes inherited field instances across models. Additionally, the | ||||
|   ordering of such fields is now defined. | ||||
|  | ||||
| .. _deprecated-features-3.2: | ||||
|  | ||||
| Features deprecated in 3.2 | ||||
|   | ||||
| @@ -102,6 +102,36 @@ class BasicFieldTests(SimpleTestCase): | ||||
|         name, path, args, kwargs = Nested.Field().deconstruct() | ||||
|         self.assertEqual(path, 'model_fields.tests.Nested.Field') | ||||
|  | ||||
|     def test_abstract_inherited_fields(self): | ||||
|         """Field instances from abstract models are not equal.""" | ||||
|         class AbstractModel(models.Model): | ||||
|             field = models.IntegerField() | ||||
|  | ||||
|             class Meta: | ||||
|                 abstract = True | ||||
|  | ||||
|         class InheritAbstractModel1(AbstractModel): | ||||
|             pass | ||||
|  | ||||
|         class InheritAbstractModel2(AbstractModel): | ||||
|             pass | ||||
|  | ||||
|         abstract_model_field = AbstractModel._meta.get_field('field') | ||||
|         inherit1_model_field = InheritAbstractModel1._meta.get_field('field') | ||||
|         inherit2_model_field = InheritAbstractModel2._meta.get_field('field') | ||||
|  | ||||
|         self.assertNotEqual(abstract_model_field, inherit1_model_field) | ||||
|         self.assertNotEqual(abstract_model_field, inherit2_model_field) | ||||
|         self.assertNotEqual(inherit1_model_field, inherit2_model_field) | ||||
|  | ||||
|         self.assertLess(abstract_model_field, inherit1_model_field) | ||||
|         self.assertLess(abstract_model_field, inherit2_model_field) | ||||
|         self.assertLess(inherit1_model_field, inherit2_model_field) | ||||
|  | ||||
|         self.assertNotEqual(hash(abstract_model_field), hash(inherit1_model_field)) | ||||
|         self.assertNotEqual(hash(abstract_model_field), hash(inherit2_model_field)) | ||||
|         self.assertNotEqual(hash(inherit1_model_field), hash(inherit2_model_field)) | ||||
|  | ||||
|  | ||||
| class ChoicesTests(SimpleTestCase): | ||||
|  | ||||
|   | ||||
		Reference in New Issue
	
	Block a user