Fixed #20577 -- Deferred filtering of prefetched related querysets.
Added internal interface to QuerySet that allows to defer next filter call till .query is accessed. Used it to optimize prefetch_related(). Thanks Simon Charette for the review.
This commit is contained in:
parent
70d8146986
commit
681f7e2b13
1
AUTHORS
1
AUTHORS
|
@ -40,6 +40,7 @@ answer newbie questions, and generally made Django that much better:
|
||||||
Alexander Dutton <dev@alexdutton.co.uk>
|
Alexander Dutton <dev@alexdutton.co.uk>
|
||||||
Alexander Myodov <alex@myodov.com>
|
Alexander Myodov <alex@myodov.com>
|
||||||
Alexandr Tatarinov <tatarinov1997@gmail.com>
|
Alexandr Tatarinov <tatarinov1997@gmail.com>
|
||||||
|
Alex Aktsipetrov <alex.akts@gmail.com>
|
||||||
Alex Becker <https://alexcbecker.net/>
|
Alex Becker <https://alexcbecker.net/>
|
||||||
Alex Couper <http://alexcouper.com/>
|
Alex Couper <http://alexcouper.com/>
|
||||||
Alex Dedul
|
Alex Dedul
|
||||||
|
|
|
@ -883,6 +883,7 @@ def create_forward_many_to_many_manager(superclass, rel, reverse):
|
||||||
queryset._add_hints(instance=self.instance)
|
queryset._add_hints(instance=self.instance)
|
||||||
if self._db:
|
if self._db:
|
||||||
queryset = queryset.using(self._db)
|
queryset = queryset.using(self._db)
|
||||||
|
queryset._defer_next_filter = True
|
||||||
return queryset._next_is_sticky().filter(**self.core_filters)
|
return queryset._next_is_sticky().filter(**self.core_filters)
|
||||||
|
|
||||||
def _remove_prefetched_objects(self):
|
def _remove_prefetched_objects(self):
|
||||||
|
|
|
@ -189,7 +189,7 @@ class QuerySet:
|
||||||
self.model = model
|
self.model = model
|
||||||
self._db = using
|
self._db = using
|
||||||
self._hints = hints or {}
|
self._hints = hints or {}
|
||||||
self.query = query or sql.Query(self.model)
|
self._query = query or sql.Query(self.model)
|
||||||
self._result_cache = None
|
self._result_cache = None
|
||||||
self._sticky_filter = False
|
self._sticky_filter = False
|
||||||
self._for_write = False
|
self._for_write = False
|
||||||
|
@ -198,6 +198,20 @@ class QuerySet:
|
||||||
self._known_related_objects = {} # {rel_field: {pk: rel_obj}}
|
self._known_related_objects = {} # {rel_field: {pk: rel_obj}}
|
||||||
self._iterable_class = ModelIterable
|
self._iterable_class = ModelIterable
|
||||||
self._fields = None
|
self._fields = None
|
||||||
|
self._defer_next_filter = False
|
||||||
|
self._deferred_filter = None
|
||||||
|
|
||||||
|
@property
|
||||||
|
def query(self):
|
||||||
|
if self._deferred_filter:
|
||||||
|
negate, args, kwargs = self._deferred_filter
|
||||||
|
self._filter_or_exclude_inplace(negate, *args, **kwargs)
|
||||||
|
self._deferred_filter = None
|
||||||
|
return self._query
|
||||||
|
|
||||||
|
@query.setter
|
||||||
|
def query(self, value):
|
||||||
|
self._query = value
|
||||||
|
|
||||||
def as_manager(cls):
|
def as_manager(cls):
|
||||||
# Address the circular dependency between `Queryset` and `Manager`.
|
# Address the circular dependency between `Queryset` and `Manager`.
|
||||||
|
@ -914,12 +928,19 @@ class QuerySet:
|
||||||
"Cannot filter a query once a slice has been taken."
|
"Cannot filter a query once a slice has been taken."
|
||||||
|
|
||||||
clone = self._chain()
|
clone = self._chain()
|
||||||
if negate:
|
if self._defer_next_filter:
|
||||||
clone.query.add_q(~Q(*args, **kwargs))
|
self._defer_next_filter = False
|
||||||
|
clone._deferred_filter = negate, args, kwargs
|
||||||
else:
|
else:
|
||||||
clone.query.add_q(Q(*args, **kwargs))
|
clone._filter_or_exclude_inplace(negate, *args, **kwargs)
|
||||||
return clone
|
return clone
|
||||||
|
|
||||||
|
def _filter_or_exclude_inplace(self, negate, *args, **kwargs):
|
||||||
|
if negate:
|
||||||
|
self._query.add_q(~Q(*args, **kwargs))
|
||||||
|
else:
|
||||||
|
self._query.add_q(Q(*args, **kwargs))
|
||||||
|
|
||||||
def complex_filter(self, filter_obj):
|
def complex_filter(self, filter_obj):
|
||||||
"""
|
"""
|
||||||
Return a new QuerySet instance with filter_obj added to the filters.
|
Return a new QuerySet instance with filter_obj added to the filters.
|
||||||
|
|
|
@ -5,6 +5,7 @@ from django.core.exceptions import ObjectDoesNotExist
|
||||||
from django.db import connection
|
from django.db import connection
|
||||||
from django.db.models import Prefetch, QuerySet
|
from django.db.models import Prefetch, QuerySet
|
||||||
from django.db.models.query import get_prefetcher, prefetch_related_objects
|
from django.db.models.query import get_prefetcher, prefetch_related_objects
|
||||||
|
from django.db.models.sql import Query
|
||||||
from django.test import TestCase, override_settings
|
from django.test import TestCase, override_settings
|
||||||
from django.test.utils import CaptureQueriesContext
|
from django.test.utils import CaptureQueriesContext
|
||||||
|
|
||||||
|
@ -291,6 +292,20 @@ class PrefetchRelatedTests(TestDataMixin, TestCase):
|
||||||
sql = queries[-1]['sql']
|
sql = queries[-1]['sql']
|
||||||
self.assertWhereContains(sql, self.author1.id)
|
self.assertWhereContains(sql, self.author1.id)
|
||||||
|
|
||||||
|
def test_filter_deferred(self):
|
||||||
|
"""
|
||||||
|
Related filtering of prefetched querysets is deferred until necessary.
|
||||||
|
"""
|
||||||
|
add_q = Query.add_q
|
||||||
|
with mock.patch.object(
|
||||||
|
Query,
|
||||||
|
'add_q',
|
||||||
|
autospec=True,
|
||||||
|
side_effect=lambda self, q: add_q(self, q),
|
||||||
|
) as add_q_mock:
|
||||||
|
list(Book.objects.prefetch_related('authors'))
|
||||||
|
self.assertEqual(add_q_mock.call_count, 1)
|
||||||
|
|
||||||
|
|
||||||
class RawQuerySetTests(TestDataMixin, TestCase):
|
class RawQuerySetTests(TestDataMixin, TestCase):
|
||||||
def test_basic(self):
|
def test_basic(self):
|
||||||
|
@ -823,6 +838,22 @@ class CustomPrefetchTests(TestCase):
|
||||||
with self.assertNumQueries(0):
|
with self.assertNumQueries(0):
|
||||||
self.assertEqual(person.cached_all_houses, all_houses)
|
self.assertEqual(person.cached_all_houses, all_houses)
|
||||||
|
|
||||||
|
def test_filter_deferred(self):
|
||||||
|
"""
|
||||||
|
Related filtering of prefetched querysets is deferred until necessary.
|
||||||
|
"""
|
||||||
|
add_q = Query.add_q
|
||||||
|
with mock.patch.object(
|
||||||
|
Query,
|
||||||
|
'add_q',
|
||||||
|
autospec=True,
|
||||||
|
side_effect=lambda self, q: add_q(self, q),
|
||||||
|
) as add_q_mock:
|
||||||
|
list(House.objects.prefetch_related(
|
||||||
|
Prefetch('occupants', queryset=Person.objects.all())
|
||||||
|
))
|
||||||
|
self.assertEqual(add_q_mock.call_count, 1)
|
||||||
|
|
||||||
|
|
||||||
class DefaultManagerTests(TestCase):
|
class DefaultManagerTests(TestCase):
|
||||||
|
|
||||||
|
|
|
@ -212,6 +212,12 @@ class PickleabilityTestCase(TestCase):
|
||||||
qs = Happening.objects.annotate(latest_time=models.Max('when'))
|
qs = Happening.objects.annotate(latest_time=models.Max('when'))
|
||||||
self.assert_pickles(qs)
|
self.assert_pickles(qs)
|
||||||
|
|
||||||
|
def test_filter_deferred(self):
|
||||||
|
qs = Happening.objects.all()
|
||||||
|
qs._defer_next_filter = True
|
||||||
|
qs = qs.filter(id=0)
|
||||||
|
self.assert_pickles(qs)
|
||||||
|
|
||||||
def test_missing_django_version_unpickling(self):
|
def test_missing_django_version_unpickling(self):
|
||||||
"""
|
"""
|
||||||
#21430 -- Verifies a warning is raised for querysets that are
|
#21430 -- Verifies a warning is raised for querysets that are
|
||||||
|
|
Loading…
Reference in New Issue