Fixed #14217 -- Added validation for field name collision when using model inheritance.

This commit is contained in:
sarthakmeh 2015-08-25 23:07:32 +05:30 committed by Tim Graham
parent 7efdd40407
commit 4bc00defd0
3 changed files with 27 additions and 1 deletions

View File

@ -629,6 +629,7 @@ answer newbie questions, and generally made Django that much better:
Ryan Niemeyer <https://profiles.google.com/ryan.niemeyer/about> Ryan Niemeyer <https://profiles.google.com/ryan.niemeyer/about>
Sam Newman <http://www.magpiebrain.com/> Sam Newman <http://www.magpiebrain.com/>
Sander Dijkhuis <sander.dijkhuis@gmail.com> Sander Dijkhuis <sander.dijkhuis@gmail.com>
Sarthak Mehrish <sarthakmeh03@gmail.com>
schwank@gmail.com schwank@gmail.com
Scot Hacker <shacker@birdhouse.org> Scot Hacker <shacker@birdhouse.org>
Scott Barr <scott@divisionbyzero.com.au> Scott Barr <scott@divisionbyzero.com.au>

View File

@ -1334,7 +1334,13 @@ class Model(six.with_metaclass(ModelBase)):
used_fields[f.attname] = f used_fields[f.attname] = f
# Check that fields defined in the model don't clash with fields from # Check that fields defined in the model don't clash with fields from
# parents. # parents, including auto-generated fields like multi-table inheritance
# child accessors.
for parent in cls._meta.get_parent_list():
for f in parent._meta.get_fields():
if f not in used_fields:
used_fields[f.name] = f
for f in cls._meta.local_fields: for f in cls._meta.local_fields:
clash = used_fields.get(f.name) or used_fields.get(f.attname) or None clash = used_fields.get(f.name) or used_fields.get(f.attname) or None
# Note that we may detect clash between user-defined non-unique # Note that we may detect clash between user-defined non-unique

View File

@ -436,6 +436,25 @@ class FieldNamesTests(IsolatedModelsTestCase):
class ShadowingFieldsTests(IsolatedModelsTestCase): class ShadowingFieldsTests(IsolatedModelsTestCase):
def test_field_name_clash_with_child_accessor(self):
class Parent(models.Model):
pass
class Child(Parent):
child = models.CharField(max_length=100)
errors = Child.check()
expected = [
Error(
"The field 'child' clashes with the field "
"'child' from model 'invalid_models_tests.parent'.",
hint=None,
obj=Child._meta.get_field('child'),
id='models.E006',
)
]
self.assertEqual(errors, expected)
def test_multiinheritance_clash(self): def test_multiinheritance_clash(self):
class Mother(models.Model): class Mother(models.Model):
clash = models.IntegerField() clash = models.IntegerField()