Fixed #18177 -- Cached known related instances.
This was recently fixed for one-to-one relations; this patch adds support for foreign keys. Thanks kaiser.yann for the report and the initial version of the patch.
This commit is contained in:
parent
3b2993ed04
commit
1e6c3368f2
|
@ -237,13 +237,18 @@ class SingleRelatedObjectDescriptor(object):
|
||||||
return self.related.model._base_manager.using(db)
|
return self.related.model._base_manager.using(db)
|
||||||
|
|
||||||
def get_prefetch_query_set(self, instances):
|
def get_prefetch_query_set(self, instances):
|
||||||
vals = set(instance._get_pk_val() for instance in instances)
|
rel_obj_attr = attrgetter(self.related.field.attname)
|
||||||
params = {'%s__pk__in' % self.related.field.name: vals}
|
instance_attr = lambda obj: obj._get_pk_val()
|
||||||
return (self.get_query_set(instance=instances[0]).filter(**params),
|
instances_dict = dict((instance_attr(inst), inst) for inst in instances)
|
||||||
attrgetter(self.related.field.attname),
|
params = {'%s__pk__in' % self.related.field.name: instances_dict.keys()}
|
||||||
lambda obj: obj._get_pk_val(),
|
qs = self.get_query_set(instance=instances[0]).filter(**params)
|
||||||
True,
|
# Since we're going to assign directly in the cache,
|
||||||
self.cache_name)
|
# we must manage the reverse relation cache manually.
|
||||||
|
rel_obj_cache_name = self.related.field.get_cache_name()
|
||||||
|
for rel_obj in qs:
|
||||||
|
instance = instances_dict[rel_obj_attr(rel_obj)]
|
||||||
|
setattr(rel_obj, rel_obj_cache_name, instance)
|
||||||
|
return qs, rel_obj_attr, instance_attr, True, self.cache_name
|
||||||
|
|
||||||
def __get__(self, instance, instance_type=None):
|
def __get__(self, instance, instance_type=None):
|
||||||
if instance is None:
|
if instance is None:
|
||||||
|
@ -324,17 +329,23 @@ class ReverseSingleRelatedObjectDescriptor(object):
|
||||||
return QuerySet(self.field.rel.to).using(db)
|
return QuerySet(self.field.rel.to).using(db)
|
||||||
|
|
||||||
def get_prefetch_query_set(self, instances):
|
def get_prefetch_query_set(self, instances):
|
||||||
vals = set(getattr(instance, self.field.attname) for instance in instances)
|
rel_obj_attr = attrgetter(self.field.rel.field_name)
|
||||||
|
instance_attr = attrgetter(self.field.attname)
|
||||||
|
instances_dict = dict((instance_attr(inst), inst) for inst in instances)
|
||||||
other_field = self.field.rel.get_related_field()
|
other_field = self.field.rel.get_related_field()
|
||||||
if other_field.rel:
|
if other_field.rel:
|
||||||
params = {'%s__pk__in' % self.field.rel.field_name: vals}
|
params = {'%s__pk__in' % self.field.rel.field_name: instances_dict.keys()}
|
||||||
else:
|
else:
|
||||||
params = {'%s__in' % self.field.rel.field_name: vals}
|
params = {'%s__in' % self.field.rel.field_name: instances_dict.keys()}
|
||||||
return (self.get_query_set(instance=instances[0]).filter(**params),
|
qs = self.get_query_set(instance=instances[0]).filter(**params)
|
||||||
attrgetter(self.field.rel.field_name),
|
# Since we're going to assign directly in the cache,
|
||||||
attrgetter(self.field.attname),
|
# we must manage the reverse relation cache manually.
|
||||||
True,
|
if not self.field.rel.multiple:
|
||||||
self.cache_name)
|
rel_obj_cache_name = self.field.related.get_cache_name()
|
||||||
|
for rel_obj in qs:
|
||||||
|
instance = instances_dict[rel_obj_attr(rel_obj)]
|
||||||
|
setattr(rel_obj, rel_obj_cache_name, instance)
|
||||||
|
return qs, rel_obj_attr, instance_attr, True, self.cache_name
|
||||||
|
|
||||||
def __get__(self, instance, instance_type=None):
|
def __get__(self, instance, instance_type=None):
|
||||||
if instance is None:
|
if instance is None:
|
||||||
|
@ -467,18 +478,24 @@ class ForeignRelatedObjectsDescriptor(object):
|
||||||
return self.instance._prefetched_objects_cache[rel_field.related_query_name()]
|
return self.instance._prefetched_objects_cache[rel_field.related_query_name()]
|
||||||
except (AttributeError, KeyError):
|
except (AttributeError, KeyError):
|
||||||
db = self._db or router.db_for_read(self.model, instance=self.instance)
|
db = self._db or router.db_for_read(self.model, instance=self.instance)
|
||||||
return super(RelatedManager, self).get_query_set().using(db).filter(**self.core_filters)
|
qs = super(RelatedManager, self).get_query_set().using(db).filter(**self.core_filters)
|
||||||
|
qs._known_related_object = (rel_field.name, self.instance)
|
||||||
|
return qs
|
||||||
|
|
||||||
def get_prefetch_query_set(self, instances):
|
def get_prefetch_query_set(self, instances):
|
||||||
|
rel_obj_attr = attrgetter(rel_field.get_attname())
|
||||||
|
instance_attr = attrgetter(attname)
|
||||||
|
instances_dict = dict((instance_attr(inst), inst) for inst in instances)
|
||||||
db = self._db or router.db_for_read(self.model, instance=instances[0])
|
db = self._db or router.db_for_read(self.model, instance=instances[0])
|
||||||
query = {'%s__%s__in' % (rel_field.name, attname):
|
query = {'%s__%s__in' % (rel_field.name, attname): instances_dict.keys()}
|
||||||
set(getattr(obj, attname) for obj in instances)}
|
|
||||||
qs = super(RelatedManager, self).get_query_set().using(db).filter(**query)
|
qs = super(RelatedManager, self).get_query_set().using(db).filter(**query)
|
||||||
return (qs,
|
# Since we just bypassed this class' get_query_set(), we must manage
|
||||||
attrgetter(rel_field.get_attname()),
|
# the reverse relation manually.
|
||||||
attrgetter(attname),
|
for rel_obj in qs:
|
||||||
False,
|
instance = instances_dict[rel_obj_attr(rel_obj)]
|
||||||
rel_field.related_query_name())
|
setattr(rel_obj, rel_field.name, instance)
|
||||||
|
cache_name = rel_field.related_query_name()
|
||||||
|
return qs, rel_obj_attr, instance_attr, False, cache_name
|
||||||
|
|
||||||
def add(self, *objs):
|
def add(self, *objs):
|
||||||
for obj in objs:
|
for obj in objs:
|
||||||
|
|
|
@ -41,6 +41,7 @@ class QuerySet(object):
|
||||||
self._for_write = False
|
self._for_write = False
|
||||||
self._prefetch_related_lookups = []
|
self._prefetch_related_lookups = []
|
||||||
self._prefetch_done = False
|
self._prefetch_done = False
|
||||||
|
self._known_related_object = None # (attname, rel_obj)
|
||||||
|
|
||||||
########################
|
########################
|
||||||
# PYTHON MAGIC METHODS #
|
# PYTHON MAGIC METHODS #
|
||||||
|
@ -282,9 +283,10 @@ class QuerySet(object):
|
||||||
init_list.append(field.attname)
|
init_list.append(field.attname)
|
||||||
model_cls = deferred_class_factory(self.model, skip)
|
model_cls = deferred_class_factory(self.model, skip)
|
||||||
|
|
||||||
# Cache db and model outside the loop
|
# Cache db, model and known_related_object outside the loop
|
||||||
db = self.db
|
db = self.db
|
||||||
model = self.model
|
model = self.model
|
||||||
|
kro_attname, kro_instance = self._known_related_object or (None, None)
|
||||||
compiler = self.query.get_compiler(using=db)
|
compiler = self.query.get_compiler(using=db)
|
||||||
if fill_cache:
|
if fill_cache:
|
||||||
klass_info = get_klass_info(model, max_depth=max_depth,
|
klass_info = get_klass_info(model, max_depth=max_depth,
|
||||||
|
@ -294,12 +296,12 @@ class QuerySet(object):
|
||||||
obj, _ = get_cached_row(row, index_start, db, klass_info,
|
obj, _ = get_cached_row(row, index_start, db, klass_info,
|
||||||
offset=len(aggregate_select))
|
offset=len(aggregate_select))
|
||||||
else:
|
else:
|
||||||
if skip:
|
# Omit aggregates in object creation.
|
||||||
row_data = row[index_start:aggregate_start]
|
row_data = row[index_start:aggregate_start]
|
||||||
|
if skip:
|
||||||
obj = model_cls(**dict(zip(init_list, row_data)))
|
obj = model_cls(**dict(zip(init_list, row_data)))
|
||||||
else:
|
else:
|
||||||
# Omit aggregates in object creation.
|
obj = model(*row_data)
|
||||||
obj = model(*row[index_start:aggregate_start])
|
|
||||||
|
|
||||||
# Store the source database of the object
|
# Store the source database of the object
|
||||||
obj._state.db = db
|
obj._state.db = db
|
||||||
|
@ -315,6 +317,10 @@ class QuerySet(object):
|
||||||
for i, aggregate in enumerate(aggregate_select):
|
for i, aggregate in enumerate(aggregate_select):
|
||||||
setattr(obj, aggregate, row[i + aggregate_start])
|
setattr(obj, aggregate, row[i + aggregate_start])
|
||||||
|
|
||||||
|
# Add the known related object to the model, if there is one
|
||||||
|
if kro_instance:
|
||||||
|
setattr(obj, kro_attname, kro_instance)
|
||||||
|
|
||||||
yield obj
|
yield obj
|
||||||
|
|
||||||
def aggregate(self, *args, **kwargs):
|
def aggregate(self, *args, **kwargs):
|
||||||
|
@ -864,6 +870,7 @@ class QuerySet(object):
|
||||||
c = klass(model=self.model, query=query, using=self._db)
|
c = klass(model=self.model, query=query, using=self._db)
|
||||||
c._for_write = self._for_write
|
c._for_write = self._for_write
|
||||||
c._prefetch_related_lookups = self._prefetch_related_lookups[:]
|
c._prefetch_related_lookups = self._prefetch_related_lookups[:]
|
||||||
|
c._known_related_object = self._known_related_object
|
||||||
c.__dict__.update(kwargs)
|
c.__dict__.update(kwargs)
|
||||||
if setup and hasattr(c, '_setup_query'):
|
if setup and hasattr(c, '_setup_query'):
|
||||||
c._setup_query()
|
c._setup_query()
|
||||||
|
@ -1781,9 +1788,7 @@ def prefetch_one_level(instances, prefetcher, attname):
|
||||||
rel_obj_cache = {}
|
rel_obj_cache = {}
|
||||||
for rel_obj in all_related_objects:
|
for rel_obj in all_related_objects:
|
||||||
rel_attr_val = rel_obj_attr(rel_obj)
|
rel_attr_val = rel_obj_attr(rel_obj)
|
||||||
if rel_attr_val not in rel_obj_cache:
|
rel_obj_cache.setdefault(rel_attr_val, []).append(rel_obj)
|
||||||
rel_obj_cache[rel_attr_val] = []
|
|
||||||
rel_obj_cache[rel_attr_val].append(rel_obj)
|
|
||||||
|
|
||||||
for obj in instances:
|
for obj in instances:
|
||||||
instance_attr_val = instance_attr(obj)
|
instance_attr_val = instance_attr(obj)
|
||||||
|
|
|
@ -44,6 +44,24 @@ reasons or when trying to avoid overwriting concurrent changes.
|
||||||
See the :meth:`Model.save() <django.db.models.Model.save()>` documentation for
|
See the :meth:`Model.save() <django.db.models.Model.save()>` documentation for
|
||||||
more details.
|
more details.
|
||||||
|
|
||||||
|
Caching of related model instances
|
||||||
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||||
|
|
||||||
|
When traversing relations, the ORM will avoid re-fetching objects that were
|
||||||
|
previously loaded. For example, with the tutorial's models::
|
||||||
|
|
||||||
|
>>> first_poll = Poll.objects.all()[0]
|
||||||
|
>>> first_choice = first_poll.choice_set.all()[0]
|
||||||
|
>>> first_choice.poll is first_poll
|
||||||
|
True
|
||||||
|
|
||||||
|
In Django 1.5, the third line no longer triggers a new SQL query to fetch
|
||||||
|
``first_choice.poll``; it was set when by the second line.
|
||||||
|
|
||||||
|
For one-to-one relationships, both sides can be cached. For many-to-one
|
||||||
|
relationships, only the single side of the relationship can be cached. This
|
||||||
|
is particularly helpful in combination with ``prefetch_related``.
|
||||||
|
|
||||||
Minor features
|
Minor features
|
||||||
~~~~~~~~~~~~~~
|
~~~~~~~~~~~~~~
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,65 @@
|
||||||
|
[
|
||||||
|
{
|
||||||
|
"pk": 1,
|
||||||
|
"model": "known_related_objects.tournament",
|
||||||
|
"fields": {
|
||||||
|
"name": "Tourney 1"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pk": 2,
|
||||||
|
"model": "known_related_objects.tournament",
|
||||||
|
"fields": {
|
||||||
|
"name": "Tourney 2"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pk": 1,
|
||||||
|
"model": "known_related_objects.pool",
|
||||||
|
"fields": {
|
||||||
|
"tournament": 1,
|
||||||
|
"name": "T1 Pool 1"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pk": 2,
|
||||||
|
"model": "known_related_objects.pool",
|
||||||
|
"fields": {
|
||||||
|
"tournament": 1,
|
||||||
|
"name": "T1 Pool 2"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pk": 3,
|
||||||
|
"model": "known_related_objects.pool",
|
||||||
|
"fields": {
|
||||||
|
"tournament": 2,
|
||||||
|
"name": "T2 Pool 1"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pk": 4,
|
||||||
|
"model": "known_related_objects.pool",
|
||||||
|
"fields": {
|
||||||
|
"tournament": 2,
|
||||||
|
"name": "T2 Pool 2"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pk": 1,
|
||||||
|
"model": "known_related_objects.poolstyle",
|
||||||
|
"fields": {
|
||||||
|
"name": "T1 Pool 2 Style",
|
||||||
|
"pool": 2
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"pk": 2,
|
||||||
|
"model": "known_related_objects.poolstyle",
|
||||||
|
"fields": {
|
||||||
|
"name": "T2 Pool 1 Style",
|
||||||
|
"pool": 3
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
|
|
@ -0,0 +1,19 @@
|
||||||
|
"""
|
||||||
|
Existing related object instance caching.
|
||||||
|
|
||||||
|
Test that queries are not redone when going back through known relations.
|
||||||
|
"""
|
||||||
|
|
||||||
|
from django.db import models
|
||||||
|
|
||||||
|
class Tournament(models.Model):
|
||||||
|
name = models.CharField(max_length=30)
|
||||||
|
|
||||||
|
class Pool(models.Model):
|
||||||
|
name = models.CharField(max_length=30)
|
||||||
|
tournament = models.ForeignKey(Tournament)
|
||||||
|
|
||||||
|
class PoolStyle(models.Model):
|
||||||
|
name = models.CharField(max_length=30)
|
||||||
|
pool = models.OneToOneField(Pool)
|
||||||
|
|
|
@ -0,0 +1,88 @@
|
||||||
|
from __future__ import absolute_import
|
||||||
|
|
||||||
|
from django.test import TestCase
|
||||||
|
|
||||||
|
from .models import Tournament, Pool, PoolStyle
|
||||||
|
|
||||||
|
class ExistingRelatedInstancesTests(TestCase):
|
||||||
|
fixtures = ['tournament.json']
|
||||||
|
|
||||||
|
def test_foreign_key(self):
|
||||||
|
with self.assertNumQueries(2):
|
||||||
|
tournament = Tournament.objects.get(pk=1)
|
||||||
|
pool = tournament.pool_set.all()[0]
|
||||||
|
self.assertIs(tournament, pool.tournament)
|
||||||
|
|
||||||
|
def test_foreign_key_prefetch_related(self):
|
||||||
|
with self.assertNumQueries(2):
|
||||||
|
tournament = (Tournament.objects.prefetch_related('pool_set').get(pk=1))
|
||||||
|
pool = tournament.pool_set.all()[0]
|
||||||
|
self.assertIs(tournament, pool.tournament)
|
||||||
|
|
||||||
|
def test_foreign_key_multiple_prefetch(self):
|
||||||
|
with self.assertNumQueries(2):
|
||||||
|
tournaments = list(Tournament.objects.prefetch_related('pool_set'))
|
||||||
|
pool1 = tournaments[0].pool_set.all()[0]
|
||||||
|
self.assertIs(tournaments[0], pool1.tournament)
|
||||||
|
pool2 = tournaments[1].pool_set.all()[0]
|
||||||
|
self.assertIs(tournaments[1], pool2.tournament)
|
||||||
|
|
||||||
|
def test_one_to_one(self):
|
||||||
|
with self.assertNumQueries(2):
|
||||||
|
style = PoolStyle.objects.get(pk=1)
|
||||||
|
pool = style.pool
|
||||||
|
self.assertIs(style, pool.poolstyle)
|
||||||
|
|
||||||
|
def test_one_to_one_select_related(self):
|
||||||
|
with self.assertNumQueries(1):
|
||||||
|
style = PoolStyle.objects.select_related('pool').get(pk=1)
|
||||||
|
pool = style.pool
|
||||||
|
self.assertIs(style, pool.poolstyle)
|
||||||
|
|
||||||
|
def test_one_to_one_multi_select_related(self):
|
||||||
|
with self.assertNumQueries(1):
|
||||||
|
poolstyles = list(PoolStyle.objects.select_related('pool'))
|
||||||
|
self.assertIs(poolstyles[0], poolstyles[0].pool.poolstyle)
|
||||||
|
self.assertIs(poolstyles[1], poolstyles[1].pool.poolstyle)
|
||||||
|
|
||||||
|
def test_one_to_one_prefetch_related(self):
|
||||||
|
with self.assertNumQueries(2):
|
||||||
|
style = PoolStyle.objects.prefetch_related('pool').get(pk=1)
|
||||||
|
pool = style.pool
|
||||||
|
self.assertIs(style, pool.poolstyle)
|
||||||
|
|
||||||
|
def test_one_to_one_multi_prefetch_related(self):
|
||||||
|
with self.assertNumQueries(2):
|
||||||
|
poolstyles = list(PoolStyle.objects.prefetch_related('pool'))
|
||||||
|
self.assertIs(poolstyles[0], poolstyles[0].pool.poolstyle)
|
||||||
|
self.assertIs(poolstyles[1], poolstyles[1].pool.poolstyle)
|
||||||
|
|
||||||
|
def test_reverse_one_to_one(self):
|
||||||
|
with self.assertNumQueries(2):
|
||||||
|
pool = Pool.objects.get(pk=2)
|
||||||
|
style = pool.poolstyle
|
||||||
|
self.assertIs(pool, style.pool)
|
||||||
|
|
||||||
|
def test_reverse_one_to_one_select_related(self):
|
||||||
|
with self.assertNumQueries(1):
|
||||||
|
pool = Pool.objects.select_related('poolstyle').get(pk=2)
|
||||||
|
style = pool.poolstyle
|
||||||
|
self.assertIs(pool, style.pool)
|
||||||
|
|
||||||
|
def test_reverse_one_to_one_prefetch_related(self):
|
||||||
|
with self.assertNumQueries(2):
|
||||||
|
pool = Pool.objects.prefetch_related('poolstyle').get(pk=2)
|
||||||
|
style = pool.poolstyle
|
||||||
|
self.assertIs(pool, style.pool)
|
||||||
|
|
||||||
|
def test_reverse_one_to_one_multi_select_related(self):
|
||||||
|
with self.assertNumQueries(1):
|
||||||
|
pools = list(Pool.objects.select_related('poolstyle'))
|
||||||
|
self.assertIs(pools[1], pools[1].poolstyle.pool)
|
||||||
|
self.assertIs(pools[2], pools[2].poolstyle.pool)
|
||||||
|
|
||||||
|
def test_reverse_one_to_one_multi_prefetch_related(self):
|
||||||
|
with self.assertNumQueries(2):
|
||||||
|
pools = list(Pool.objects.prefetch_related('poolstyle'))
|
||||||
|
self.assertIs(pools[1], pools[1].poolstyle.pool)
|
||||||
|
self.assertIs(pools[2], pools[2].poolstyle.pool)
|
Loading…
Reference in New Issue