2017-05-06 02:19:34 +00:00
|
|
|
import operator
|
2016-06-03 22:31:21 +00:00
|
|
|
import unittest
|
|
|
|
from collections import namedtuple
|
2017-05-06 02:19:34 +00:00
|
|
|
from contextlib import contextmanager
|
2016-06-03 22:31:21 +00:00
|
|
|
|
2017-11-19 15:13:10 +00:00
|
|
|
from django.db import connection, models
|
2016-06-03 22:31:21 +00:00
|
|
|
from django.test import TestCase
|
|
|
|
|
2017-02-11 20:37:49 +00:00
|
|
|
from ..models import Person
|
2016-06-03 22:31:21 +00:00
|
|
|
|
|
|
|
|
2017-02-11 20:37:49 +00:00
|
|
|
@unittest.skipUnless(connection.vendor == 'postgresql', 'PostgreSQL tests')
|
2016-06-03 22:31:21 +00:00
|
|
|
class ServerSideCursorsPostgres(TestCase):
|
|
|
|
cursor_fields = 'name, statement, is_holdable, is_binary, is_scrollable, creation_time'
|
|
|
|
PostgresCursor = namedtuple('PostgresCursor', cursor_fields)
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def setUpTestData(cls):
|
|
|
|
Person.objects.create(first_name='a', last_name='a')
|
|
|
|
Person.objects.create(first_name='b', last_name='b')
|
|
|
|
|
|
|
|
def inspect_cursors(self):
|
|
|
|
with connection.cursor() as cursor:
|
|
|
|
cursor.execute('SELECT {fields} FROM pg_cursors;'.format(fields=self.cursor_fields))
|
|
|
|
cursors = cursor.fetchall()
|
|
|
|
return [self.PostgresCursor._make(cursor) for cursor in cursors]
|
|
|
|
|
2017-05-06 02:19:34 +00:00
|
|
|
@contextmanager
|
|
|
|
def override_db_setting(self, **kwargs):
|
2017-12-06 22:17:59 +00:00
|
|
|
for setting in kwargs:
|
2017-05-06 02:19:34 +00:00
|
|
|
original_value = connection.settings_dict.get(setting)
|
|
|
|
if setting in connection.settings_dict:
|
|
|
|
self.addCleanup(operator.setitem, connection.settings_dict, setting, original_value)
|
|
|
|
else:
|
|
|
|
self.addCleanup(operator.delitem, connection.settings_dict, setting)
|
|
|
|
|
|
|
|
connection.settings_dict[setting] = kwargs[setting]
|
|
|
|
yield
|
|
|
|
|
2017-11-21 13:11:50 +00:00
|
|
|
def assertUsesCursor(self, queryset, num_expected=1):
|
|
|
|
next(queryset) # Open a server-side cursor
|
2016-06-03 22:31:21 +00:00
|
|
|
cursors = self.inspect_cursors()
|
2017-11-21 13:11:50 +00:00
|
|
|
self.assertEqual(len(cursors), num_expected)
|
2016-06-03 22:31:21 +00:00
|
|
|
for cursor in cursors:
|
|
|
|
self.assertIn('_django_curs_', cursor.name)
|
|
|
|
self.assertFalse(cursor.is_scrollable)
|
|
|
|
self.assertFalse(cursor.is_holdable)
|
|
|
|
self.assertFalse(cursor.is_binary)
|
|
|
|
|
2017-11-21 13:11:50 +00:00
|
|
|
def asserNotUsesCursor(self, queryset):
|
|
|
|
self.assertUsesCursor(queryset, num_expected=0)
|
|
|
|
|
|
|
|
def test_server_side_cursor(self):
|
|
|
|
self.assertUsesCursor(Person.objects.iterator())
|
|
|
|
|
2017-11-19 15:13:10 +00:00
|
|
|
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())
|
|
|
|
|
2017-11-21 13:11:50 +00:00
|
|
|
def test_server_side_cursor_many_cursors(self):
|
|
|
|
persons = Person.objects.iterator()
|
|
|
|
persons2 = Person.objects.iterator()
|
|
|
|
next(persons) # Open a server-side cursor
|
|
|
|
self.assertUsesCursor(persons2, num_expected=2)
|
|
|
|
|
2016-06-03 22:31:21 +00:00
|
|
|
def test_closed_server_side_cursor(self):
|
|
|
|
persons = Person.objects.iterator()
|
|
|
|
next(persons) # Open a server-side cursor
|
|
|
|
del persons
|
|
|
|
cursors = self.inspect_cursors()
|
|
|
|
self.assertEqual(len(cursors), 0)
|
2017-05-06 02:19:34 +00:00
|
|
|
|
|
|
|
def test_server_side_cursors_setting(self):
|
|
|
|
with self.override_db_setting(DISABLE_SERVER_SIDE_CURSORS=False):
|
|
|
|
persons = Person.objects.iterator()
|
2017-11-21 13:11:50 +00:00
|
|
|
self.assertUsesCursor(persons)
|
2017-05-06 02:19:34 +00:00
|
|
|
del persons # Close server-side cursor
|
|
|
|
|
|
|
|
with self.override_db_setting(DISABLE_SERVER_SIDE_CURSORS=True):
|
2017-11-21 13:11:50 +00:00
|
|
|
self.asserNotUsesCursor(Person.objects.iterator())
|