mirror of
				https://github.com/django/django.git
				synced 2025-10-25 14:46:09 +00:00 
			
		
		
		
	Fixed #26327 -- Added JsonAgg to contrib.postgres.
Thanks Tim Graham for review.
This commit is contained in:
		| @@ -1,7 +1,8 @@ | |||||||
|  | from django.contrib.postgres.fields import JSONField | ||||||
| from django.db.models.aggregates import Aggregate | from django.db.models.aggregates import Aggregate | ||||||
|  |  | ||||||
| __all__ = [ | __all__ = [ | ||||||
|     'ArrayAgg', 'BitAnd', 'BitOr', 'BoolAnd', 'BoolOr', 'StringAgg', |     'ArrayAgg', 'BitAnd', 'BitOr', 'BoolAnd', 'BoolOr', 'JsonAgg', 'StringAgg', | ||||||
| ] | ] | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -30,6 +31,16 @@ class BoolOr(Aggregate): | |||||||
|     function = 'BOOL_OR' |     function = 'BOOL_OR' | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class JsonAgg(Aggregate): | ||||||
|  |     function = 'JSONB_AGG' | ||||||
|  |     _output_field = JSONField() | ||||||
|  |  | ||||||
|  |     def convert_value(self, value, expression, connection, context): | ||||||
|  |         if not value: | ||||||
|  |             return [] | ||||||
|  |         return value | ||||||
|  |  | ||||||
|  |  | ||||||
| class StringAgg(Aggregate): | class StringAgg(Aggregate): | ||||||
|     function = 'STRING_AGG' |     function = 'STRING_AGG' | ||||||
|     template = "%(function)s(%(distinct)s%(expressions)s, '%(delimiter)s')" |     template = "%(function)s(%(distinct)s%(expressions)s, '%(delimiter)s')" | ||||||
|   | |||||||
| @@ -58,6 +58,15 @@ General-purpose aggregation functions | |||||||
|     Returns ``True`` if at least one input value is true, ``None`` if all |     Returns ``True`` if at least one input value is true, ``None`` if all | ||||||
|     values are null or if there are no values, otherwise ``False``. |     values are null or if there are no values, otherwise ``False``. | ||||||
|  |  | ||||||
|  | ``JsonAgg`` | ||||||
|  | ----------- | ||||||
|  |  | ||||||
|  | .. class:: JsonAgg(expressions, **extra) | ||||||
|  |  | ||||||
|  |     .. versionadded:: 1.11 | ||||||
|  |  | ||||||
|  |     Returns the input values as a ``JSON`` array. | ||||||
|  |  | ||||||
| ``StringAgg`` | ``StringAgg`` | ||||||
| ------------- | ------------- | ||||||
|  |  | ||||||
|   | |||||||
| @@ -172,6 +172,9 @@ Minor features | |||||||
|   operation allow using PostgreSQL's ``citext`` extension for case-insensitive |   operation allow using PostgreSQL's ``citext`` extension for case-insensitive | ||||||
|   lookups. |   lookups. | ||||||
|  |  | ||||||
|  | * The new :class:`~django.contrib.postgres.aggregates.JsonAgg` allows | ||||||
|  |   aggregating values as a JSON array. | ||||||
|  |  | ||||||
| :mod:`django.contrib.redirects` | :mod:`django.contrib.redirects` | ||||||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,14 +1,21 @@ | |||||||
| from django.contrib.postgres.aggregates import ( | import json | ||||||
|     ArrayAgg, BitAnd, BitOr, BoolAnd, BoolOr, Corr, CovarPop, RegrAvgX, |  | ||||||
|     RegrAvgY, RegrCount, RegrIntercept, RegrR2, RegrSlope, RegrSXX, RegrSXY, |  | ||||||
|     RegrSYY, StatAggregate, StringAgg, |  | ||||||
| ) |  | ||||||
| from django.db.models.expressions import F, Value | from django.db.models.expressions import F, Value | ||||||
|  | from django.test.testcases import skipUnlessDBFeature | ||||||
| from django.test.utils import Approximate | from django.test.utils import Approximate | ||||||
|  |  | ||||||
| from . import PostgreSQLTestCase | from . import PostgreSQLTestCase | ||||||
| from .models import AggregateTestModel, StatTestModel | from .models import AggregateTestModel, StatTestModel | ||||||
|  |  | ||||||
|  | try: | ||||||
|  |     from django.contrib.postgres.aggregates import ( | ||||||
|  |         ArrayAgg, BitAnd, BitOr, BoolAnd, BoolOr, Corr, CovarPop, JsonAgg, | ||||||
|  |         RegrAvgX, RegrAvgY, RegrCount, RegrIntercept, RegrR2, RegrSlope, | ||||||
|  |         RegrSXX, RegrSXY, RegrSYY, StatAggregate, StringAgg, | ||||||
|  |     ) | ||||||
|  | except ImportError: | ||||||
|  |     pass  # psycopg2 is not installed | ||||||
|  |  | ||||||
|  |  | ||||||
| class TestGeneralAggregate(PostgreSQLTestCase): | class TestGeneralAggregate(PostgreSQLTestCase): | ||||||
|     @classmethod |     @classmethod | ||||||
| @@ -110,6 +117,16 @@ class TestGeneralAggregate(PostgreSQLTestCase): | |||||||
|         values = AggregateTestModel.objects.aggregate(stringagg=StringAgg('char_field', delimiter=';')) |         values = AggregateTestModel.objects.aggregate(stringagg=StringAgg('char_field', delimiter=';')) | ||||||
|         self.assertEqual(values, {'stringagg': ''}) |         self.assertEqual(values, {'stringagg': ''}) | ||||||
|  |  | ||||||
|  |     @skipUnlessDBFeature('has_jsonb_datatype') | ||||||
|  |     def test_json_agg(self): | ||||||
|  |         values = AggregateTestModel.objects.aggregate(jsonagg=JsonAgg('char_field')) | ||||||
|  |         self.assertEqual(values, {'jsonagg': ['Foo1', 'Foo2', 'Foo3', 'Foo4']}) | ||||||
|  |  | ||||||
|  |     @skipUnlessDBFeature('has_jsonb_datatype') | ||||||
|  |     def test_json_agg_empty(self): | ||||||
|  |         values = AggregateTestModel.objects.none().aggregate(jsonagg=JsonAgg('integer_field')) | ||||||
|  |         self.assertEqual(values, json.loads('{"jsonagg": []}')) | ||||||
|  |  | ||||||
|  |  | ||||||
| class TestStringAggregateDistinct(PostgreSQLTestCase): | class TestStringAggregateDistinct(PostgreSQLTestCase): | ||||||
|     @classmethod |     @classmethod | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user