Fixed #32089 -- Fixed prefetch_related_objects() when some objects are already fetched.

Thanks Dennis Kliban for the report and Adam Johnson for the initial
patch.

Co-authored-by: Adam Johnson <me@adamj.eu>
This commit is contained in:
Alexey Nigin 2020-12-13 19:02:45 -05:00 committed by Mariusz Felisiak
parent bebd4cfa8f
commit b9ba85a7ce
2 changed files with 52 additions and 10 deletions

View File

@ -1720,8 +1720,17 @@ def prefetch_related_objects(model_instances, *related_lookups):
"prefetching - this is an invalid parameter to " "prefetching - this is an invalid parameter to "
"prefetch_related()." % lookup.prefetch_through) "prefetch_related()." % lookup.prefetch_through)
if prefetcher is not None and not is_fetched: obj_to_fetch = None
obj_list, additional_lookups = prefetch_one_level(obj_list, prefetcher, lookup, level) if prefetcher is not None:
obj_to_fetch = [obj for obj in obj_list if not is_fetched(obj)]
if obj_to_fetch:
obj_list, additional_lookups = prefetch_one_level(
obj_to_fetch,
prefetcher,
lookup,
level,
)
# We need to ensure we don't keep adding lookups from the # We need to ensure we don't keep adding lookups from the
# same relationships to stop infinite recursion. So, if we # same relationships to stop infinite recursion. So, if we
# are already on an automatically added lookup, don't add # are already on an automatically added lookup, don't add
@ -1771,10 +1780,14 @@ def get_prefetcher(instance, through_attr, to_attr):
(the object with get_prefetch_queryset (or None), (the object with get_prefetch_queryset (or None),
the descriptor object representing this relationship (or None), the descriptor object representing this relationship (or None),
a boolean that is False if the attribute was not found at all, a boolean that is False if the attribute was not found at all,
a boolean that is True if the attribute has already been fetched) a function that takes an instance and returns a boolean that is True if
the attribute has already been fetched for that instance)
""" """
def has_to_attr_attribute(instance):
return hasattr(instance, to_attr)
prefetcher = None prefetcher = None
is_fetched = False is_fetched = has_to_attr_attribute
# For singly related objects, we have to avoid getting the attribute # For singly related objects, we have to avoid getting the attribute
# from the object, as this will trigger the query. So we first try # from the object, as this will trigger the query. So we first try
@ -1789,8 +1802,7 @@ def get_prefetcher(instance, through_attr, to_attr):
# get_prefetch_queryset() method. # get_prefetch_queryset() method.
if hasattr(rel_obj_descriptor, 'get_prefetch_queryset'): if hasattr(rel_obj_descriptor, 'get_prefetch_queryset'):
prefetcher = rel_obj_descriptor prefetcher = rel_obj_descriptor
if rel_obj_descriptor.is_cached(instance): is_fetched = rel_obj_descriptor.is_cached
is_fetched = True
else: else:
# descriptor doesn't support prefetching, so we go ahead and get # descriptor doesn't support prefetching, so we go ahead and get
# the attribute on the instance rather than the class to # the attribute on the instance rather than the class to
@ -1802,11 +1814,15 @@ def get_prefetcher(instance, through_attr, to_attr):
# Special case cached_property instances because hasattr # Special case cached_property instances because hasattr
# triggers attribute computation and assignment. # triggers attribute computation and assignment.
if isinstance(getattr(instance.__class__, to_attr, None), cached_property): if isinstance(getattr(instance.__class__, to_attr, None), cached_property):
is_fetched = to_attr in instance.__dict__ def has_cached_property(instance):
else: return to_attr in instance.__dict__
is_fetched = hasattr(instance, to_attr)
is_fetched = has_cached_property
else: else:
is_fetched = through_attr in instance._prefetched_objects_cache def in_prefetched_cache(instance):
return through_attr in instance._prefetched_objects_cache
is_fetched = in_prefetched_cache
return prefetcher, rel_obj_descriptor, attr_found, is_fetched return prefetcher, rel_obj_descriptor, attr_found, is_fetched

View File

@ -97,6 +97,16 @@ class PrefetchRelatedObjectsTests(TestCase):
with self.assertNumQueries(0): with self.assertNumQueries(0):
self.assertCountEqual(book1.authors.all(), [self.author1, self.author2, self.author3]) self.assertCountEqual(book1.authors.all(), [self.author1, self.author2, self.author3])
def test_prefetch_object_twice(self):
book1 = Book.objects.get(id=self.book1.id)
book2 = Book.objects.get(id=self.book2.id)
with self.assertNumQueries(1):
prefetch_related_objects([book1], Prefetch('authors'))
with self.assertNumQueries(1):
prefetch_related_objects([book1, book2], Prefetch('authors'))
with self.assertNumQueries(0):
self.assertCountEqual(book2.authors.all(), [self.author1])
def test_prefetch_object_to_attr(self): def test_prefetch_object_to_attr(self):
book1 = Book.objects.get(id=self.book1.id) book1 = Book.objects.get(id=self.book1.id)
with self.assertNumQueries(1): with self.assertNumQueries(1):
@ -105,6 +115,22 @@ class PrefetchRelatedObjectsTests(TestCase):
with self.assertNumQueries(0): with self.assertNumQueries(0):
self.assertCountEqual(book1.the_authors, [self.author1, self.author2, self.author3]) self.assertCountEqual(book1.the_authors, [self.author1, self.author2, self.author3])
def test_prefetch_object_to_attr_twice(self):
book1 = Book.objects.get(id=self.book1.id)
book2 = Book.objects.get(id=self.book2.id)
with self.assertNumQueries(1):
prefetch_related_objects(
[book1],
Prefetch('authors', to_attr='the_authors'),
)
with self.assertNumQueries(1):
prefetch_related_objects(
[book1, book2],
Prefetch('authors', to_attr='the_authors'),
)
with self.assertNumQueries(0):
self.assertCountEqual(book2.the_authors, [self.author1])
def test_prefetch_queryset(self): def test_prefetch_queryset(self):
book1 = Book.objects.get(id=self.book1.id) book1 = Book.objects.get(id=self.book1.id)
with self.assertNumQueries(1): with self.assertNumQueries(1):