Skip to content

Commit

Permalink
Fixed #28817 -- Made QuerySet.iterator() use server-side cursors afte…
Browse files Browse the repository at this point in the history
…r values() and values_list().
  • Loading branch information
dodobas authored and timgraham committed Nov 21, 2017
1 parent 6cb6382 commit d97f026
Show file tree
Hide file tree
Showing 4 changed files with 27 additions and 7 deletions.
11 changes: 7 additions & 4 deletions django/db/models/query.py
Original file line number Diff line number Diff line change
Expand Up @@ -105,7 +105,7 @@ def __iter__(self):
names = extra_names + field_names + annotation_names

indexes = range(len(names))
for row in compiler.results_iter():
for row in compiler.results_iter(chunked_fetch=self.chunked_fetch, chunk_size=self.chunk_size):
yield {names[i]: row[i] for i in indexes}


Expand Down Expand Up @@ -133,8 +133,11 @@ def __iter__(self):
# Reorder according to fields.
index_map = {name: idx for idx, name in enumerate(names)}
rowfactory = operator.itemgetter(*[index_map[f] for f in fields])
return map(rowfactory, compiler.results_iter())
return compiler.results_iter(tuple_expected=True)
return map(
rowfactory,
compiler.results_iter(chunked_fetch=self.chunked_fetch, chunk_size=self.chunk_size)
)
return compiler.results_iter(tuple_expected=True, chunked_fetch=self.chunked_fetch, chunk_size=self.chunk_size)


class NamedValuesListIterable(ValuesListIterable):
Expand Down Expand Up @@ -174,7 +177,7 @@ class FlatValuesListIterable(BaseIterable):
def __iter__(self):
queryset = self.queryset
compiler = queryset.query.get_compiler(queryset.db)
return chain.from_iterable(compiler.results_iter())
return chain.from_iterable(compiler.results_iter(chunked_fetch=self.chunked_fetch, chunk_size=self.chunk_size))


class QuerySet:
Expand Down
5 changes: 3 additions & 2 deletions django/db/models/sql/compiler.py
Original file line number Diff line number Diff line change
Expand Up @@ -1002,10 +1002,11 @@ def apply_converters(self, rows, converters):
row[pos] = value
yield row

def results_iter(self, results=None, tuple_expected=False):
def results_iter(self, results=None, tuple_expected=False, chunked_fetch=False,
chunk_size=GET_ITERATOR_CHUNK_SIZE):
"""Return an iterator over the results from executing this query."""
if results is None:
results = self.execute_sql(MULTI)
results = self.execute_sql(MULTI, chunked_fetch=chunked_fetch, chunk_size=chunk_size)
fields = [s[0] for s in self.select[0:self.col_count]]
converters = self.get_converters(fields)
rows = chain.from_iterable(results)
Expand Down
3 changes: 3 additions & 0 deletions docs/releases/1.11.8.txt
Original file line number Diff line number Diff line change
Expand Up @@ -18,3 +18,6 @@ Bugfixes

* Fixed incorrect index name truncation when using a namespaced ``db_table``
(:ticket:`28792`).

* Made ``QuerySet.iterator()`` use server-side cursors on PostgreSQL after
``values()`` and ``values_list()`` (:ticket:`28817`).
15 changes: 14 additions & 1 deletion tests/backends/postgresql/test_server_side_cursors.py
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@
from collections import namedtuple
from contextlib import contextmanager

from django.db import connection
from django.db import connection, models
from django.test import TestCase

from ..models import Person
Expand Down Expand Up @@ -53,6 +53,19 @@ def asserNotUsesCursor(self, queryset):
def test_server_side_cursor(self):
self.assertUsesCursor(Person.objects.iterator())

def test_values(self):
self.assertUsesCursor(Person.objects.values('first_name').iterator())

def test_values_list(self):
self.assertUsesCursor(Person.objects.values_list('first_name').iterator())

def test_values_list_flat(self):
self.assertUsesCursor(Person.objects.values_list('first_name', flat=True).iterator())

def test_values_list_fields_not_equal_to_names(self):
expr = models.Count('id')
self.assertUsesCursor(Person.objects.annotate(id__count=expr).values_list(expr, 'id__count').iterator())

def test_server_side_cursor_many_cursors(self):
persons = Person.objects.iterator()
persons2 = Person.objects.iterator()
Expand Down

0 comments on commit d97f026

Please sign in to comment.