[1.0.X] Fixed #10251 -- Fixed model inheritance when there's also an explicit pk field.
Backport of r9970 from trunk. git-svn-id: http://code.djangoproject.com/svn/django/branches/releases/1.0.X@9972 bcc190cf-cafb-0310-a4f2-bffc1f526a37
This commit is contained in:
parent
b57d86f226
commit
dce458cde2
|
@ -436,6 +436,21 @@ class Options(object):
|
||||||
result.update(parent._meta.get_parent_list())
|
result.update(parent._meta.get_parent_list())
|
||||||
return result
|
return result
|
||||||
|
|
||||||
|
def get_ancestor_link(self, ancestor):
|
||||||
|
"""
|
||||||
|
Returns the field on the current model which points to the given
|
||||||
|
"ancestor". This is possible an indirect link (a pointer to a parent
|
||||||
|
model, which points, eventually, to the ancestor). Used when
|
||||||
|
constructing table joins for model inheritance.
|
||||||
|
|
||||||
|
Returns None if the model isn't an ancestor of this one.
|
||||||
|
"""
|
||||||
|
if ancestor in self.parents:
|
||||||
|
return self.parents[ancestor]
|
||||||
|
for parent in self.parents:
|
||||||
|
if parent._meta.get_ancestor_link(ancestor):
|
||||||
|
return self.parents[parent]
|
||||||
|
|
||||||
def get_ordered_objects(self):
|
def get_ordered_objects(self):
|
||||||
"Returns a list of Options objects that are ordered with respect to this object."
|
"Returns a list of Options objects that are ordered with respect to this object."
|
||||||
if not hasattr(self, '_ordered_objects'):
|
if not hasattr(self, '_ordered_objects'):
|
||||||
|
|
|
@ -494,14 +494,14 @@ class BaseQuery(object):
|
||||||
aliases = set()
|
aliases = set()
|
||||||
if start_alias:
|
if start_alias:
|
||||||
seen = {None: start_alias}
|
seen = {None: start_alias}
|
||||||
root_pk = opts.pk.column
|
|
||||||
for field, model in opts.get_fields_with_model():
|
for field, model in opts.get_fields_with_model():
|
||||||
if start_alias:
|
if start_alias:
|
||||||
try:
|
try:
|
||||||
alias = seen[model]
|
alias = seen[model]
|
||||||
except KeyError:
|
except KeyError:
|
||||||
|
link_field = opts.get_ancestor_link(model)
|
||||||
alias = self.join((start_alias, model._meta.db_table,
|
alias = self.join((start_alias, model._meta.db_table,
|
||||||
root_pk, model._meta.pk.column))
|
link_field.column, model._meta.pk.column))
|
||||||
seen[model] = alias
|
seen[model] = alias
|
||||||
else:
|
else:
|
||||||
# If we're starting from the base model of the queryset, the
|
# If we're starting from the base model of the queryset, the
|
||||||
|
@ -1005,13 +1005,13 @@ class BaseQuery(object):
|
||||||
as_sql()).
|
as_sql()).
|
||||||
"""
|
"""
|
||||||
opts = self.model._meta
|
opts = self.model._meta
|
||||||
root_pk = opts.pk.column
|
|
||||||
root_alias = self.tables[0]
|
root_alias = self.tables[0]
|
||||||
seen = {None: root_alias}
|
seen = {None: root_alias}
|
||||||
for field, model in opts.get_fields_with_model():
|
for field, model in opts.get_fields_with_model():
|
||||||
if model not in seen:
|
if model not in seen:
|
||||||
|
link_field = opts.get_ancestor_link(model)
|
||||||
seen[model] = self.join((root_alias, model._meta.db_table,
|
seen[model] = self.join((root_alias, model._meta.db_table,
|
||||||
root_pk, model._meta.pk.column))
|
link_field.column, model._meta.pk.column))
|
||||||
self.included_inherited_models = seen
|
self.included_inherited_models = seen
|
||||||
|
|
||||||
def remove_inherited_models(self):
|
def remove_inherited_models(self):
|
||||||
|
|
|
@ -86,6 +86,19 @@ class Evaluation(Article):
|
||||||
class QualityControl(Evaluation):
|
class QualityControl(Evaluation):
|
||||||
assignee = models.CharField(max_length=50)
|
assignee = models.CharField(max_length=50)
|
||||||
|
|
||||||
|
class BaseM(models.Model):
|
||||||
|
base_name = models.CharField(max_length=100)
|
||||||
|
|
||||||
|
def __unicode__(self):
|
||||||
|
return self.base_name
|
||||||
|
|
||||||
|
class DerivedM(BaseM):
|
||||||
|
customPK = models.IntegerField(primary_key=True)
|
||||||
|
derived_name = models.CharField(max_length=100)
|
||||||
|
|
||||||
|
def __unicode__(self):
|
||||||
|
return "PK = %d, base_name = %s, derived_name = %s" \
|
||||||
|
% (self.customPK, self.base_name, self.derived_name)
|
||||||
|
|
||||||
__test__ = {'API_TESTS':"""
|
__test__ = {'API_TESTS':"""
|
||||||
# Regression for #7350, #7202
|
# Regression for #7350, #7202
|
||||||
|
@ -275,4 +288,10 @@ True
|
||||||
>>> ArticleWithAuthor.objects.filter(pk=article.pk).update(headline="Oh, no!")
|
>>> ArticleWithAuthor.objects.filter(pk=article.pk).update(headline="Oh, no!")
|
||||||
1
|
1
|
||||||
|
|
||||||
|
>>> DerivedM.objects.create(customPK=44, base_name="b1", derived_name="d1")
|
||||||
|
<DerivedM: PK = 44, base_name = b1, derived_name = d1>
|
||||||
|
>>> DerivedM.objects.all()
|
||||||
|
[<DerivedM: PK = 44, base_name = b1, derived_name = d1>]
|
||||||
|
|
||||||
"""}
|
"""}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue