mirror of
				https://github.com/django/django.git
				synced 2025-10-24 22:26:08 +00:00 
			
		
		
		
	Refs #14357 -- Deprecated Meta.ordering affecting GROUP BY queries.
Thanks Ramiro Morales for contributing to the patch.
This commit is contained in:
		
				
					committed by
					
						 Tim Graham
						Tim Graham
					
				
			
			
				
	
			
			
			
						parent
						
							c52ecbda61
						
					
				
				
					commit
					1b1f64ee5a
				
			| @@ -14,7 +14,9 @@ from django.db.models.sql.constants import ( | |||||||
| from django.db.models.sql.query import Query, get_order_dir | from django.db.models.sql.query import Query, get_order_dir | ||||||
| from django.db.transaction import TransactionManagementError | from django.db.transaction import TransactionManagementError | ||||||
| from django.db.utils import DatabaseError, NotSupportedError | from django.db.utils import DatabaseError, NotSupportedError | ||||||
| from django.utils.deprecation import RemovedInDjango30Warning | from django.utils.deprecation import ( | ||||||
|  |     RemovedInDjango30Warning, RemovedInDjango31Warning, | ||||||
|  | ) | ||||||
| from django.utils.inspect import func_supports_parameter | from django.utils.inspect import func_supports_parameter | ||||||
|  |  | ||||||
| FORCE = object() | FORCE = object() | ||||||
| @@ -34,6 +36,7 @@ class SQLCompiler: | |||||||
|         self.annotation_col_map = None |         self.annotation_col_map = None | ||||||
|         self.klass_info = None |         self.klass_info = None | ||||||
|         self.ordering_parts = re.compile(r'(.*)\s(ASC|DESC)(.*)') |         self.ordering_parts = re.compile(r'(.*)\s(ASC|DESC)(.*)') | ||||||
|  |         self._meta_ordering = None | ||||||
|  |  | ||||||
|     def setup_query(self): |     def setup_query(self): | ||||||
|         if all(self.query.alias_refcount[a] == 0 for a in self.query.alias_map): |         if all(self.query.alias_refcount[a] == 0 for a in self.query.alias_map): | ||||||
| @@ -266,8 +269,13 @@ class SQLCompiler: | |||||||
|             ordering = self.query.extra_order_by |             ordering = self.query.extra_order_by | ||||||
|         elif not self.query.default_ordering: |         elif not self.query.default_ordering: | ||||||
|             ordering = self.query.order_by |             ordering = self.query.order_by | ||||||
|  |         elif self.query.order_by: | ||||||
|  |             ordering = self.query.order_by | ||||||
|  |         elif self.query.get_meta().ordering: | ||||||
|  |             ordering = self.query.get_meta().ordering | ||||||
|  |             self._meta_ordering = ordering | ||||||
|         else: |         else: | ||||||
|             ordering = (self.query.order_by or self.query.get_meta().ordering or []) |             ordering = [] | ||||||
|         if self.query.standard_ordering: |         if self.query.standard_ordering: | ||||||
|             asc, desc = ORDER_DIR['ASC'] |             asc, desc = ORDER_DIR['ASC'] | ||||||
|         else: |         else: | ||||||
| @@ -536,7 +544,18 @@ class SQLCompiler: | |||||||
|                         raise NotImplementedError('annotate() + distinct(fields) is not implemented.') |                         raise NotImplementedError('annotate() + distinct(fields) is not implemented.') | ||||||
|                     order_by = order_by or self.connection.ops.force_no_ordering() |                     order_by = order_by or self.connection.ops.force_no_ordering() | ||||||
|                     result.append('GROUP BY %s' % ', '.join(grouping)) |                     result.append('GROUP BY %s' % ', '.join(grouping)) | ||||||
|  |                     if self._meta_ordering: | ||||||
|  |                         # When the deprecation ends, replace with: | ||||||
|  |                         # order_by = None | ||||||
|  |                         warnings.warn( | ||||||
|  |                             "%s QuerySet won't use Meta.ordering in Django 3.1. " | ||||||
|  |                             "Add .order_by('%s') to retain the current query." % ( | ||||||
|  |                                 self.query.model.__name__, | ||||||
|  |                                 "', '".join(self._meta_ordering) | ||||||
|  |                             ), | ||||||
|  |                             RemovedInDjango31Warning, | ||||||
|  |                             stacklevel=4, | ||||||
|  |                         ) | ||||||
|                 if having: |                 if having: | ||||||
|                     result.append('HAVING %s' % having) |                     result.append('HAVING %s' % having) | ||||||
|                     params.extend(h_params) |                     params.extend(h_params) | ||||||
|   | |||||||
| @@ -19,6 +19,8 @@ details on these changes. | |||||||
|  |  | ||||||
| * ``django.core.paginator.QuerySetPaginator`` will be removed. | * ``django.core.paginator.QuerySetPaginator`` will be removed. | ||||||
|  |  | ||||||
|  | * A model's ``Meta.ordering`` will no longer affect ``GROUP BY`` queries. | ||||||
|  |  | ||||||
| .. _deprecation-removed-in-3.0: | .. _deprecation-removed-in-3.0: | ||||||
|  |  | ||||||
| 3.0 | 3.0 | ||||||
|   | |||||||
| @@ -285,7 +285,8 @@ Django quotes column and table names behind the scenes. | |||||||
|         ordering = [F('author').asc(nulls_last=True)] |         ordering = [F('author').asc(nulls_last=True)] | ||||||
|  |  | ||||||
|     Default ordering also affects :ref:`aggregation queries |     Default ordering also affects :ref:`aggregation queries | ||||||
|     <aggregation-ordering-interaction>`. |     <aggregation-ordering-interaction>` but this won't be the case starting | ||||||
|  |     in Django 3.1. | ||||||
|  |  | ||||||
| .. warning:: | .. warning:: | ||||||
|  |  | ||||||
|   | |||||||
| @@ -289,6 +289,15 @@ Miscellaneous | |||||||
| Features deprecated in 2.2 | Features deprecated in 2.2 | ||||||
| ========================== | ========================== | ||||||
|  |  | ||||||
|  | Model ``Meta.ordering`` will no longer affect ``GROUP BY`` queries | ||||||
|  | ------------------------------------------------------------------ | ||||||
|  |  | ||||||
|  | A model's ``Meta.ordering`` affecting ``GROUP BY`` queries (such as | ||||||
|  | ``.annotate().values()``) is a common source of confusion. Such queries now | ||||||
|  | issue a deprecation warning with the advice to add an ``order_by()`` to retain | ||||||
|  | the current query. ``Meta.ordering`` will be ignored in such queries starting | ||||||
|  | in Django 3.1. | ||||||
|  |  | ||||||
| Miscellaneous | Miscellaneous | ||||||
| ------------- | ------------- | ||||||
|  |  | ||||||
|   | |||||||
| @@ -514,6 +514,13 @@ include the aggregate column. | |||||||
| Interaction with default ordering or ``order_by()`` | Interaction with default ordering or ``order_by()`` | ||||||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | ||||||
|  |  | ||||||
|  | .. deprecated:: 2.2 | ||||||
|  |  | ||||||
|  |     Starting in Django 3.1, the ordering from a model's ``Meta.ordering`` won't | ||||||
|  |     be used in ``GROUP BY`` queries, such as ``.annotate().values()``. Since | ||||||
|  |     Django 2.2, these queries issue a deprecation warning indicating to add an | ||||||
|  |     explicit ``order_by()`` to the queryset to silence the warning. | ||||||
|  |  | ||||||
| Fields that are mentioned in the ``order_by()`` part of a queryset (or which | Fields that are mentioned in the ``order_by()`` part of a queryset (or which | ||||||
| are used in the default ordering on a model) are used when selecting the | are used in the default ordering on a model) are used when selecting the | ||||||
| output data, even if they are not otherwise specified in the ``values()`` | output data, even if they are not otherwise specified in the ``values()`` | ||||||
|   | |||||||
| @@ -11,8 +11,11 @@ from django.db.models import ( | |||||||
|     Avg, Case, Count, DecimalField, F, IntegerField, Max, Q, StdDev, Sum, |     Avg, Case, Count, DecimalField, F, IntegerField, Max, Q, StdDev, Sum, | ||||||
|     Value, Variance, When, |     Value, Variance, When, | ||||||
| ) | ) | ||||||
| from django.test import TestCase, skipUnlessAnyDBFeature, skipUnlessDBFeature | from django.test import ( | ||||||
|  |     TestCase, ignore_warnings, skipUnlessAnyDBFeature, skipUnlessDBFeature, | ||||||
|  | ) | ||||||
| from django.test.utils import Approximate | from django.test.utils import Approximate | ||||||
|  | from django.utils.deprecation import RemovedInDjango31Warning | ||||||
|  |  | ||||||
| from .models import ( | from .models import ( | ||||||
|     Alfa, Author, Book, Bravo, Charlie, Clues, Entries, HardbackBook, ItemTag, |     Alfa, Author, Book, Bravo, Charlie, Clues, Entries, HardbackBook, ItemTag, | ||||||
| @@ -106,6 +109,7 @@ class AggregationTests(TestCase): | |||||||
|         for attr, value in kwargs.items(): |         for attr, value in kwargs.items(): | ||||||
|             self.assertEqual(getattr(obj, attr), value) |             self.assertEqual(getattr(obj, attr), value) | ||||||
|  |  | ||||||
|  |     @ignore_warnings(category=RemovedInDjango31Warning) | ||||||
|     def test_annotation_with_value(self): |     def test_annotation_with_value(self): | ||||||
|         values = Book.objects.filter( |         values = Book.objects.filter( | ||||||
|             name='Practical Django Projects', |             name='Practical Django Projects', | ||||||
| @@ -213,6 +217,7 @@ class AggregationTests(TestCase): | |||||||
|             {'pages__sum': 3703} |             {'pages__sum': 3703} | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|  |     @ignore_warnings(category=RemovedInDjango31Warning) | ||||||
|     def test_annotation(self): |     def test_annotation(self): | ||||||
|         # Annotations get combined with extra select clauses |         # Annotations get combined with extra select clauses | ||||||
|         obj = Book.objects.annotate(mean_auth_age=Avg("authors__age")).extra( |         obj = Book.objects.annotate(mean_auth_age=Avg("authors__age")).extra( | ||||||
| @@ -306,6 +311,7 @@ class AggregationTests(TestCase): | |||||||
|  |  | ||||||
|         # If an annotation isn't included in the values, it can still be used |         # If an annotation isn't included in the values, it can still be used | ||||||
|         # in a filter |         # in a filter | ||||||
|  |         with ignore_warnings(category=RemovedInDjango31Warning): | ||||||
|             qs = Book.objects.annotate(n_authors=Count('authors')).values('name').filter(n_authors__gt=2) |             qs = Book.objects.annotate(n_authors=Count('authors')).values('name').filter(n_authors__gt=2) | ||||||
|         self.assertSequenceEqual( |         self.assertSequenceEqual( | ||||||
|             qs, [ |             qs, [ | ||||||
| @@ -450,6 +456,7 @@ class AggregationTests(TestCase): | |||||||
|         with self.assertRaisesMessage(FieldError, msg): |         with self.assertRaisesMessage(FieldError, msg): | ||||||
|             Book.objects.all().annotate(num_authors=Count('authors__id')).aggregate(Max('foo')) |             Book.objects.all().annotate(num_authors=Count('authors__id')).aggregate(Max('foo')) | ||||||
|  |  | ||||||
|  |     @ignore_warnings(category=RemovedInDjango31Warning) | ||||||
|     def test_more(self): |     def test_more(self): | ||||||
|         # Old-style count aggregations can be mixed with new-style |         # Old-style count aggregations can be mixed with new-style | ||||||
|         self.assertEqual( |         self.assertEqual( | ||||||
| @@ -679,7 +686,7 @@ class AggregationTests(TestCase): | |||||||
|         ) |         ) | ||||||
|  |  | ||||||
|         # Regression for #10127 - Empty select_related() works with annotate |         # Regression for #10127 - Empty select_related() works with annotate | ||||||
|         qs = Book.objects.filter(rating__lt=4.5).select_related().annotate(Avg('authors__age')) |         qs = Book.objects.filter(rating__lt=4.5).select_related().annotate(Avg('authors__age')).order_by('name') | ||||||
|         self.assertQuerysetEqual( |         self.assertQuerysetEqual( | ||||||
|             qs, |             qs, | ||||||
|             [ |             [ | ||||||
| @@ -803,6 +810,7 @@ class AggregationTests(TestCase): | |||||||
|         with self.assertRaisesMessage(ValueError, msg): |         with self.assertRaisesMessage(ValueError, msg): | ||||||
|             Author.objects.annotate(book_contact_set=Avg('friends__age')) |             Author.objects.annotate(book_contact_set=Avg('friends__age')) | ||||||
|  |  | ||||||
|  |     @ignore_warnings(category=RemovedInDjango31Warning) | ||||||
|     def test_pickle(self): |     def test_pickle(self): | ||||||
|         # Regression for #10197 -- Queries with aggregates can be pickled. |         # Regression for #10197 -- Queries with aggregates can be pickled. | ||||||
|         # First check that pickling is possible at all. No crash = success |         # First check that pickling is possible at all. No crash = success | ||||||
| @@ -933,7 +941,9 @@ class AggregationTests(TestCase): | |||||||
|             {'n_pages': 2078}, |             {'n_pages': 2078}, | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|         qs = HardbackBook.objects.annotate(n_authors=Count('book_ptr__authors')).values('name', 'n_authors') |         qs = HardbackBook.objects.annotate( | ||||||
|  |             n_authors=Count('book_ptr__authors'), | ||||||
|  |         ).values('name', 'n_authors').order_by('name') | ||||||
|         self.assertSequenceEqual( |         self.assertSequenceEqual( | ||||||
|             qs, |             qs, | ||||||
|             [ |             [ | ||||||
| @@ -945,7 +955,7 @@ class AggregationTests(TestCase): | |||||||
|             ], |             ], | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|         qs = HardbackBook.objects.annotate(n_authors=Count('authors')).values('name', 'n_authors') |         qs = HardbackBook.objects.annotate(n_authors=Count('authors')).values('name', 'n_authors').order_by('name') | ||||||
|         self.assertSequenceEqual( |         self.assertSequenceEqual( | ||||||
|             qs, |             qs, | ||||||
|             [ |             [ | ||||||
| @@ -1005,7 +1015,7 @@ class AggregationTests(TestCase): | |||||||
|     def test_values_annotate_values(self): |     def test_values_annotate_values(self): | ||||||
|         qs = Book.objects.values("name").annotate( |         qs = Book.objects.values("name").annotate( | ||||||
|             n_authors=Count("authors") |             n_authors=Count("authors") | ||||||
|         ).values_list("pk", flat=True) |         ).values_list("pk", flat=True).order_by('name') | ||||||
|         self.assertEqual(list(qs), list(Book.objects.values_list("pk", flat=True))) |         self.assertEqual(list(qs), list(Book.objects.values_list("pk", flat=True))) | ||||||
|  |  | ||||||
|     def test_having_group_by(self): |     def test_having_group_by(self): | ||||||
| @@ -1015,7 +1025,7 @@ class AggregationTests(TestCase): | |||||||
|             n_authors=Count("authors") |             n_authors=Count("authors") | ||||||
|         ).filter( |         ).filter( | ||||||
|             pages__gt=F("n_authors") |             pages__gt=F("n_authors") | ||||||
|         ).values_list("name", flat=True) |         ).values_list("name", flat=True).order_by('name') | ||||||
|         # Results should be the same, all Books have more pages than authors |         # Results should be the same, all Books have more pages than authors | ||||||
|         self.assertEqual( |         self.assertEqual( | ||||||
|             list(qs), list(Book.objects.values_list("name", flat=True)) |             list(qs), list(Book.objects.values_list("name", flat=True)) | ||||||
| @@ -1035,7 +1045,7 @@ class AggregationTests(TestCase): | |||||||
|     def test_annotation_disjunction(self): |     def test_annotation_disjunction(self): | ||||||
|         qs = Book.objects.annotate(n_authors=Count("authors")).filter( |         qs = Book.objects.annotate(n_authors=Count("authors")).filter( | ||||||
|             Q(n_authors=2) | Q(name="Python Web Development with Django") |             Q(n_authors=2) | Q(name="Python Web Development with Django") | ||||||
|         ) |         ).order_by('name') | ||||||
|         self.assertQuerysetEqual( |         self.assertQuerysetEqual( | ||||||
|             qs, [ |             qs, [ | ||||||
|                 "Artificial Intelligence: A Modern Approach", |                 "Artificial Intelligence: A Modern Approach", | ||||||
| @@ -1052,7 +1062,7 @@ class AggregationTests(TestCase): | |||||||
|                 Q(name="The Definitive Guide to Django: Web Development Done Right") | |                 Q(name="The Definitive Guide to Django: Web Development Done Right") | | ||||||
|                 (Q(name="Artificial Intelligence: A Modern Approach") & Q(n_authors=3)) |                 (Q(name="Artificial Intelligence: A Modern Approach") & Q(n_authors=3)) | ||||||
|             ) |             ) | ||||||
|         ) |         ).order_by('name') | ||||||
|         self.assertQuerysetEqual( |         self.assertQuerysetEqual( | ||||||
|             qs, |             qs, | ||||||
|             [ |             [ | ||||||
| @@ -1200,6 +1210,7 @@ class AggregationTests(TestCase): | |||||||
|             {'book__count__max': 2} |             {'book__count__max': 2} | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|  |     @ignore_warnings(category=RemovedInDjango31Warning) | ||||||
|     def test_annotate_joins(self): |     def test_annotate_joins(self): | ||||||
|         """ |         """ | ||||||
|         The base table's join isn't promoted to LOUTER. This could |         The base table's join isn't promoted to LOUTER. This could | ||||||
| @@ -1436,6 +1447,7 @@ class AggregationTests(TestCase): | |||||||
|         query.filter(q1 | q2) |         query.filter(q1 | q2) | ||||||
|         self.assertEqual(len(q2.children), 1) |         self.assertEqual(len(q2.children), 1) | ||||||
|  |  | ||||||
|  |     @ignore_warnings(category=RemovedInDjango31Warning) | ||||||
|     def test_fobj_group_by(self): |     def test_fobj_group_by(self): | ||||||
|         """ |         """ | ||||||
|         An F() object referring to related column works correctly in group by. |         An F() object referring to related column works correctly in group by. | ||||||
| @@ -1513,6 +1525,7 @@ class JoinPromotionTests(TestCase): | |||||||
|         qs = Charlie.objects.annotate(Count('alfa__name')) |         qs = Charlie.objects.annotate(Count('alfa__name')) | ||||||
|         self.assertIn(' LEFT OUTER JOIN ', str(qs.query)) |         self.assertIn(' LEFT OUTER JOIN ', str(qs.query)) | ||||||
|  |  | ||||||
|  |     @ignore_warnings(category=RemovedInDjango31Warning) | ||||||
|     def test_non_nullable_fk_not_promoted(self): |     def test_non_nullable_fk_not_promoted(self): | ||||||
|         qs = Book.objects.annotate(Count('contact__name')) |         qs = Book.objects.annotate(Count('contact__name')) | ||||||
|         self.assertIn(' INNER JOIN ', str(qs.query)) |         self.assertIn(' INNER JOIN ', str(qs.query)) | ||||||
|   | |||||||
| @@ -1,9 +1,10 @@ | |||||||
| from datetime import datetime | from datetime import datetime | ||||||
| from operator import attrgetter | from operator import attrgetter | ||||||
|  |  | ||||||
| from django.db.models import DateTimeField, F, Max, OuterRef, Subquery | from django.db.models import Count, DateTimeField, F, Max, OuterRef, Subquery | ||||||
| from django.db.models.functions import Upper | from django.db.models.functions import Upper | ||||||
| from django.test import TestCase | from django.test import TestCase | ||||||
|  | from django.utils.deprecation import RemovedInDjango31Warning | ||||||
|  |  | ||||||
| from .models import Article, Author, OrderedByFArticle, Reference | from .models import Article, Author, OrderedByFArticle, Reference | ||||||
|  |  | ||||||
| @@ -403,3 +404,11 @@ class OrderingTests(TestCase): | |||||||
|             articles, ['Article 1', 'Article 4', 'Article 3', 'Article 2'], |             articles, ['Article 1', 'Article 4', 'Article 3', 'Article 2'], | ||||||
|             attrgetter('headline') |             attrgetter('headline') | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|  |     def test_deprecated_values_annotate(self): | ||||||
|  |         msg = ( | ||||||
|  |             "Article QuerySet won't use Meta.ordering in Django 3.1. Add " | ||||||
|  |             ".order_by('-pub_date', 'headline') to retain the current query." | ||||||
|  |         ) | ||||||
|  |         with self.assertRaisesMessage(RemovedInDjango31Warning, msg): | ||||||
|  |             list(Article.objects.values('author').annotate(Count('headline'))) | ||||||
|   | |||||||
| @@ -2,8 +2,11 @@ import unittest | |||||||
|  |  | ||||||
| from django.db import NotSupportedError, connection, transaction | from django.db import NotSupportedError, connection, transaction | ||||||
| from django.db.models import Count | from django.db.models import Count | ||||||
| from django.test import TestCase, skipIfDBFeature, skipUnlessDBFeature | from django.test import ( | ||||||
|  |     TestCase, ignore_warnings, skipIfDBFeature, skipUnlessDBFeature, | ||||||
|  | ) | ||||||
| from django.test.utils import CaptureQueriesContext | from django.test.utils import CaptureQueriesContext | ||||||
|  | from django.utils.deprecation import RemovedInDjango31Warning | ||||||
|  |  | ||||||
| from .models import Tag | from .models import Tag | ||||||
|  |  | ||||||
| @@ -11,6 +14,7 @@ from .models import Tag | |||||||
| @skipUnlessDBFeature('supports_explaining_query_execution') | @skipUnlessDBFeature('supports_explaining_query_execution') | ||||||
| class ExplainTests(TestCase): | class ExplainTests(TestCase): | ||||||
|  |  | ||||||
|  |     @ignore_warnings(category=RemovedInDjango31Warning) | ||||||
|     def test_basic(self): |     def test_basic(self): | ||||||
|         querysets = [ |         querysets = [ | ||||||
|             Tag.objects.filter(name='test'), |             Tag.objects.filter(name='test'), | ||||||
|   | |||||||
| @@ -1156,7 +1156,7 @@ class Queries1Tests(TestCase): | |||||||
|     def test_ticket_20250(self): |     def test_ticket_20250(self): | ||||||
|         # A negated Q along with an annotated queryset failed in Django 1.4 |         # A negated Q along with an annotated queryset failed in Django 1.4 | ||||||
|         qs = Author.objects.annotate(Count('item')) |         qs = Author.objects.annotate(Count('item')) | ||||||
|         qs = qs.filter(~Q(extra__value=0)) |         qs = qs.filter(~Q(extra__value=0)).order_by('name') | ||||||
|  |  | ||||||
|         self.assertIn('SELECT', str(qs.query)) |         self.assertIn('SELECT', str(qs.query)) | ||||||
|         self.assertQuerysetEqual( |         self.assertQuerysetEqual( | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user