Fixed #7246 -- Pull in the all the necessary data when using select_related() with multi-table inheritance.
git-svn-id: http://code.djangoproject.com/svn/django/trunk@7781 bcc190cf-cafb-0310-a4f2-bffc1f526a37
This commit is contained in:
parent
b67164fb59
commit
a8fa3fd81f
|
@ -442,28 +442,39 @@ class Query(object):
|
||||||
self._select_aliases = aliases
|
self._select_aliases = aliases
|
||||||
return result
|
return result
|
||||||
|
|
||||||
def get_default_columns(self, with_aliases=False, col_aliases=None):
|
def get_default_columns(self, with_aliases=False, col_aliases=None,
|
||||||
|
start_alias=None, opts=None, as_pairs=False):
|
||||||
"""
|
"""
|
||||||
Computes the default columns for selecting every field in the base
|
Computes the default columns for selecting every field in the base
|
||||||
model.
|
model.
|
||||||
|
|
||||||
Returns a list of strings, quoted appropriately for use in SQL
|
Returns a list of strings, quoted appropriately for use in SQL
|
||||||
directly, as well as a set of aliases used in the select statement.
|
directly, as well as a set of aliases used in the select statement (if
|
||||||
|
'as_pairs' is True, returns a list of (alias, col_name) pairs instead
|
||||||
|
of strings as the first component and None as the second component).
|
||||||
"""
|
"""
|
||||||
result = []
|
result = []
|
||||||
|
if opts is None:
|
||||||
|
opts = self.model._meta
|
||||||
|
if start_alias:
|
||||||
|
table_alias = start_alias
|
||||||
|
else:
|
||||||
table_alias = self.tables[0]
|
table_alias = self.tables[0]
|
||||||
root_pk = self.model._meta.pk.column
|
root_pk = opts.pk.column
|
||||||
seen = {None: table_alias}
|
seen = {None: table_alias}
|
||||||
qn = self.quote_name_unless_alias
|
qn = self.quote_name_unless_alias
|
||||||
qn2 = self.connection.ops.quote_name
|
qn2 = self.connection.ops.quote_name
|
||||||
aliases = set()
|
aliases = set()
|
||||||
for field, model in self.model._meta.get_fields_with_model():
|
for field, model in opts.get_fields_with_model():
|
||||||
try:
|
try:
|
||||||
alias = seen[model]
|
alias = seen[model]
|
||||||
except KeyError:
|
except KeyError:
|
||||||
alias = self.join((table_alias, model._meta.db_table,
|
alias = self.join((table_alias, model._meta.db_table,
|
||||||
root_pk, model._meta.pk.column))
|
root_pk, model._meta.pk.column))
|
||||||
seen[model] = alias
|
seen[model] = alias
|
||||||
|
if as_pairs:
|
||||||
|
result.append((alias, field.column))
|
||||||
|
continue
|
||||||
if with_aliases and field.column in col_aliases:
|
if with_aliases and field.column in col_aliases:
|
||||||
c_alias = 'Col%d' % len(col_aliases)
|
c_alias = 'Col%d' % len(col_aliases)
|
||||||
result.append('%s.%s AS %s' % (qn(alias),
|
result.append('%s.%s AS %s' % (qn(alias),
|
||||||
|
@ -476,6 +487,8 @@ class Query(object):
|
||||||
aliases.add(r)
|
aliases.add(r)
|
||||||
if with_aliases:
|
if with_aliases:
|
||||||
col_aliases.add(field.column)
|
col_aliases.add(field.column)
|
||||||
|
if as_pairs:
|
||||||
|
return result, None
|
||||||
return result, aliases
|
return result, aliases
|
||||||
|
|
||||||
def get_from_clause(self):
|
def get_from_clause(self):
|
||||||
|
@ -941,8 +954,8 @@ class Query(object):
|
||||||
f.rel.get_related_field().column), exclusions=used,
|
f.rel.get_related_field().column), exclusions=used,
|
||||||
promote=promote)
|
promote=promote)
|
||||||
used.add(alias)
|
used.add(alias)
|
||||||
self.related_select_cols.extend([(alias, f2.column)
|
self.related_select_cols.extend(self.get_default_columns(
|
||||||
for f2 in f.rel.to._meta.fields])
|
start_alias=alias, opts=f.rel.to._meta, as_pairs=True)[0])
|
||||||
self.related_select_fields.extend(f.rel.to._meta.fields)
|
self.related_select_fields.extend(f.rel.to._meta.fields)
|
||||||
if restricted:
|
if restricted:
|
||||||
next = requested.get(f.name, {})
|
next = requested.get(f.name, {})
|
||||||
|
|
|
@ -0,0 +1,48 @@
|
||||||
|
"""
|
||||||
|
Regression tests for the interaction between model inheritance and
|
||||||
|
select_related().
|
||||||
|
"""
|
||||||
|
|
||||||
|
from django.db import models
|
||||||
|
|
||||||
|
class Place(models.Model):
|
||||||
|
name = models.CharField(max_length=50)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
ordering = ('name',)
|
||||||
|
|
||||||
|
def __unicode__(self):
|
||||||
|
return u"%s the place" % self.name
|
||||||
|
|
||||||
|
class Restaurant(Place):
|
||||||
|
serves_sushi = models.BooleanField()
|
||||||
|
serves_steak = models.BooleanField()
|
||||||
|
|
||||||
|
def __unicode__(self):
|
||||||
|
return u"%s the restaurant" % self.name
|
||||||
|
|
||||||
|
class Person(models.Model):
|
||||||
|
name = models.CharField(max_length=50)
|
||||||
|
favorite_restaurant = models.ForeignKey(Restaurant)
|
||||||
|
|
||||||
|
def __unicode__(self):
|
||||||
|
return self.name
|
||||||
|
|
||||||
|
__test__ = {'API_TESTS':"""
|
||||||
|
Regression test for #7246
|
||||||
|
|
||||||
|
>>> r1 = Restaurant.objects.create(name="Nobu", serves_sushi=True, serves_steak=False)
|
||||||
|
>>> r2 = Restaurant.objects.create(name="Craft", serves_sushi=False, serves_steak=True)
|
||||||
|
>>> p1 = Person.objects.create(name="John", favorite_restaurant=r1)
|
||||||
|
>>> p2 = Person.objects.create(name="Jane", favorite_restaurant=r2)
|
||||||
|
|
||||||
|
>>> Person.objects.order_by('name').select_related()
|
||||||
|
[<Person: Jane>, <Person: John>]
|
||||||
|
|
||||||
|
>>> Person.objects.order_by('name').select_related('favorite_restaurant').query.as_sql()
|
||||||
|
>>> jane = Person.objects.order_by('name').select_related('favorite_restaurant')[0]
|
||||||
|
>>> jane.favorite_restaurant.name
|
||||||
|
u'Craft'
|
||||||
|
|
||||||
|
"""}
|
||||||
|
|
Loading…
Reference in New Issue