Fixed #29847 -- Ensured proper ordering in queries.

Even though good databases tend to keep the result sorted by the/one
window expression and the planners are smart enough to not resort if not
required, it is not valid to rely on this.

MariaDB specifically did return whatever order it wanted, which is
completely okay. Now we sort towards the expected data for all databases.
This commit is contained in:
Florian Apolloner 2018-10-21 20:42:34 +02:00 committed by Florian Apolloner
parent e127ef62de
commit c53af56613
1 changed files with 4 additions and 14 deletions

View File

@ -16,13 +16,6 @@ from django.test import SimpleTestCase, TestCase, skipUnlessDBFeature
from .models import Employee from .models import Employee
def fix_ordering_for_mariadb(qs, ordering):
if connection.vendor == 'mysql' and connection.mysql_is_mariadb:
# MariaDB requires repeating the ordering when using window functions
qs = qs.order_by(*ordering)
return qs
@skipUnlessDBFeature('supports_over_clause') @skipUnlessDBFeature('supports_over_clause')
class WindowFunctionTests(TestCase): class WindowFunctionTests(TestCase):
@classmethod @classmethod
@ -193,8 +186,7 @@ class WindowFunctionTests(TestCase):
expression=Lag(expression='salary', offset=1), expression=Lag(expression='salary', offset=1),
partition_by=F('department'), partition_by=F('department'),
order_by=[F('salary').asc(), F('name').asc()], order_by=[F('salary').asc(), F('name').asc()],
)).order_by('department') )).order_by('department', F('salary').asc(), F('name').asc())
qs = fix_ordering_for_mariadb(qs, ('department', F('salary').asc(), F('name').asc()))
self.assertQuerysetEqual(qs, [ self.assertQuerysetEqual(qs, [
('Williams', 37000, 'Accounting', None), ('Williams', 37000, 'Accounting', None),
('Jenson', 45000, 'Accounting', 37000), ('Jenson', 45000, 'Accounting', 37000),
@ -257,8 +249,8 @@ class WindowFunctionTests(TestCase):
expression=Lead(expression='salary'), expression=Lead(expression='salary'),
order_by=[F('hire_date').asc(), F('name').desc()], order_by=[F('hire_date').asc(), F('name').desc()],
partition_by='department', partition_by='department',
)).values_list('name', 'salary', 'department', 'hire_date', 'lead') )).values_list('name', 'salary', 'department', 'hire_date', 'lead') \
qs = fix_ordering_for_mariadb(qs, ('department', F('hire_date').asc(), F('name').desc())) .order_by('department', F('hire_date').asc(), F('name').desc())
self.assertNotIn('GROUP BY', str(qs.query)) self.assertNotIn('GROUP BY', str(qs.query))
self.assertSequenceEqual(qs, [ self.assertSequenceEqual(qs, [
('Jones', 45000, 'Accounting', datetime.date(2005, 11, 1), 45000), ('Jones', 45000, 'Accounting', datetime.date(2005, 11, 1), 45000),
@ -381,9 +373,7 @@ class WindowFunctionTests(TestCase):
expression=Lead(expression='salary'), expression=Lead(expression='salary'),
order_by=[F('hire_date').asc(), F('name').desc()], order_by=[F('hire_date').asc(), F('name').desc()],
partition_by='department', partition_by='department',
)).order_by('department') )).order_by('department', F('hire_date').asc(), F('name').desc())
('department', F('hire_date').asc(), F('name').desc())
qs = fix_ordering_for_mariadb(qs, ('department', F('hire_date').asc(), F('name').desc()))
self.assertQuerysetEqual(qs, [ self.assertQuerysetEqual(qs, [
('Jones', 45000, 'Accounting', datetime.date(2005, 11, 1), 45000), ('Jones', 45000, 'Accounting', datetime.date(2005, 11, 1), 45000),
('Jenson', 45000, 'Accounting', datetime.date(2008, 4, 1), 37000), ('Jenson', 45000, 'Accounting', datetime.date(2008, 4, 1), 37000),