mirror of
				https://github.com/django/django.git
				synced 2025-10-25 22:56:12 +00:00 
			
		
		
		
	[4.2.x] Fixed #34987 -- Fixed queryset crash when mixing aggregate and window annotations.
Regression inf387d024fc. Just like `OrderByList` the `ExpressionList` expression used to wrap `Window.partition_by` must implement `get_group_by_cols` to ensure the necessary grouping when mixing window expressions with aggregate annotations is performed against the partition members and not the partition expression itself. This is necessary because while `partition_by` is implemented as a source expression of `Window` it's actually a fragment of the WINDOW expression at the SQL level and thus it should result in a group by its members and not the sum of them. Thanks ElRoberto538 for the report. Backport ofe76cc93b01from main
This commit is contained in:
		
				
					committed by
					
						 Mariusz Felisiak
						Mariusz Felisiak
					
				
			
			
				
	
			
			
			
						parent
						
							6d7313bc87
						
					
				
				
					commit
					cf95de9d24
				
			| @@ -1223,6 +1223,12 @@ class ExpressionList(Func): | |||||||
|         # Casting to numeric is unnecessary. |         # Casting to numeric is unnecessary. | ||||||
|         return self.as_sql(compiler, connection, **extra_context) |         return self.as_sql(compiler, connection, **extra_context) | ||||||
|  |  | ||||||
|  |     def get_group_by_cols(self): | ||||||
|  |         group_by_cols = [] | ||||||
|  |         for partition in self.get_source_expressions(): | ||||||
|  |             group_by_cols.extend(partition.get_group_by_cols()) | ||||||
|  |         return group_by_cols | ||||||
|  |  | ||||||
|  |  | ||||||
| class OrderByList(Func): | class OrderByList(Func): | ||||||
|     template = "ORDER BY %(expressions)s" |     template = "ORDER BY %(expressions)s" | ||||||
|   | |||||||
| @@ -16,3 +16,7 @@ Bugfixes | |||||||
| * Fixed a regression in Django 4.2 that caused a crash of | * Fixed a regression in Django 4.2 that caused a crash of | ||||||
|   ``QuerySet.aggregate()`` with aggregates referencing other aggregates or |   ``QuerySet.aggregate()`` with aggregates referencing other aggregates or | ||||||
|   window functions through conditional expressions (:ticket:`34975`). |   window functions through conditional expressions (:ticket:`34975`). | ||||||
|  |  | ||||||
|  | * Fixed a regression in Django 4.2 that caused a crash when annotating a | ||||||
|  |   ``QuerySet`` with a ``Window`` expressions composed of a ``partition_by`` | ||||||
|  |   clause mixing field types and aggregation expressions (:ticket:`34987`). | ||||||
|   | |||||||
| @@ -837,23 +837,24 @@ class WindowFunctionTests(TestCase): | |||||||
|             max=Window( |             max=Window( | ||||||
|                 expression=Max("salary"), |                 expression=Max("salary"), | ||||||
|                 partition_by=[F("department"), F("hire_date__year")], |                 partition_by=[F("department"), F("hire_date__year")], | ||||||
|             ) |             ), | ||||||
|  |             past_department_count=Count("past_departments"), | ||||||
|         ).order_by("department", "hire_date", "name") |         ).order_by("department", "hire_date", "name") | ||||||
|         self.assertQuerySetEqual( |         self.assertQuerySetEqual( | ||||||
|             qs, |             qs, | ||||||
|             [ |             [ | ||||||
|                 ("Jones", 45000, "Accounting", datetime.date(2005, 11, 1), 45000), |                 ("Jones", 45000, "Accounting", datetime.date(2005, 11, 1), 45000, 0), | ||||||
|                 ("Jenson", 45000, "Accounting", datetime.date(2008, 4, 1), 45000), |                 ("Jenson", 45000, "Accounting", datetime.date(2008, 4, 1), 45000, 0), | ||||||
|                 ("Williams", 37000, "Accounting", datetime.date(2009, 6, 1), 37000), |                 ("Williams", 37000, "Accounting", datetime.date(2009, 6, 1), 37000, 0), | ||||||
|                 ("Adams", 50000, "Accounting", datetime.date(2013, 7, 1), 50000), |                 ("Adams", 50000, "Accounting", datetime.date(2013, 7, 1), 50000, 0), | ||||||
|                 ("Wilkinson", 60000, "IT", datetime.date(2011, 3, 1), 60000), |                 ("Wilkinson", 60000, "IT", datetime.date(2011, 3, 1), 60000, 0), | ||||||
|                 ("Moore", 34000, "IT", datetime.date(2013, 8, 1), 34000), |                 ("Moore", 34000, "IT", datetime.date(2013, 8, 1), 34000, 0), | ||||||
|                 ("Miller", 100000, "Management", datetime.date(2005, 6, 1), 100000), |                 ("Miller", 100000, "Management", datetime.date(2005, 6, 1), 100000, 1), | ||||||
|                 ("Johnson", 80000, "Management", datetime.date(2005, 7, 1), 100000), |                 ("Johnson", 80000, "Management", datetime.date(2005, 7, 1), 100000, 0), | ||||||
|                 ("Smith", 38000, "Marketing", datetime.date(2009, 10, 1), 38000), |                 ("Smith", 38000, "Marketing", datetime.date(2009, 10, 1), 38000, 0), | ||||||
|                 ("Johnson", 40000, "Marketing", datetime.date(2012, 3, 1), 40000), |                 ("Johnson", 40000, "Marketing", datetime.date(2012, 3, 1), 40000, 1), | ||||||
|                 ("Smith", 55000, "Sales", datetime.date(2007, 6, 1), 55000), |                 ("Smith", 55000, "Sales", datetime.date(2007, 6, 1), 55000, 0), | ||||||
|                 ("Brown", 53000, "Sales", datetime.date(2009, 9, 1), 53000), |                 ("Brown", 53000, "Sales", datetime.date(2009, 9, 1), 53000, 0), | ||||||
|             ], |             ], | ||||||
|             transform=lambda row: ( |             transform=lambda row: ( | ||||||
|                 row.name, |                 row.name, | ||||||
| @@ -861,6 +862,7 @@ class WindowFunctionTests(TestCase): | |||||||
|                 row.department, |                 row.department, | ||||||
|                 row.hire_date, |                 row.hire_date, | ||||||
|                 row.max, |                 row.max, | ||||||
|  |                 row.past_department_count, | ||||||
|             ), |             ), | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user