From: Mike Bayer Date: Wed, 22 Aug 2018 15:13:54 +0000 (-0400) Subject: Pass desired array type from pg.array_agg to functions.array_agg X-Git-Tag: rel_1_3_0b1~98 X-Git-Url: http://git.ipfire.org/cgi-bin/gitweb.cgi?a=commitdiff_plain;h=52a3f5b7635583ae6feb084b1db654b9c65caec2;p=thirdparty%2Fsqlalchemy%2Fsqlalchemy.git Pass desired array type from pg.array_agg to functions.array_agg Fixed the :func:`.postgresql.array_agg` function, which is a slightly altered version of the usual :func:`.functions.array_agg` function, to also accept an incoming "type" argument without forcing an ARRAY around it, essentially the same thing that was fixed for the generic function in 1.1 in :ticket:`4107`. Fixes: #4324 Change-Id: I399a29f59c945a217cdd22c65ff0325edea8ea65 --- diff --git a/doc/build/changelog/unreleased_12/4324.rst b/doc/build/changelog/unreleased_12/4324.rst new file mode 100644 index 0000000000..5f3e879038 --- /dev/null +++ b/doc/build/changelog/unreleased_12/4324.rst @@ -0,0 +1,9 @@ +.. change:: + :tags: bug, postgresql + :tickets: 4324 + + Fixed the :func:`.postgresql.array_agg` function, which is a slightly + altered version of the usual :func:`.functions.array_agg` function, to also + accept an incoming "type" argument without forcing an ARRAY around it, + essentially the same thing that was fixed for the generic function in 1.1 + in :ticket:`4107`. diff --git a/lib/sqlalchemy/dialects/postgresql/ext.py b/lib/sqlalchemy/dialects/postgresql/ext.py index 20ed0fc8d6..71fb3cc5bc 100644 --- a/lib/sqlalchemy/dialects/postgresql/ext.py +++ b/lib/sqlalchemy/dialects/postgresql/ext.py @@ -209,10 +209,11 @@ static/sql-createtable.html#SQL-CREATETABLE-EXCLUDE def array_agg(*arg, **kw): """PostgreSQL-specific form of :class:`.array_agg`, ensures return type is :class:`.postgresql.ARRAY` and not - the plain :class:`.types.ARRAY`. + the plain :class:`.types.ARRAY`, unless an explicit ``type_`` + is passed. .. versionadded:: 1.1 """ - kw['type_'] = ARRAY(functions._type_from_args(arg)) + kw['_default_array_type'] = ARRAY return functions.func.array_agg(*arg, **kw) diff --git a/lib/sqlalchemy/sql/functions.py b/lib/sqlalchemy/sql/functions.py index 27d030d4ff..5cea7750a7 100644 --- a/lib/sqlalchemy/sql/functions.py +++ b/lib/sqlalchemy/sql/functions.py @@ -793,13 +793,14 @@ class array_agg(GenericFunction): def __init__(self, *args, **kwargs): args = [_literal_as_binds(c) for c in args] + default_array_type = kwargs.pop('_default_array_type', sqltypes.ARRAY) if 'type_' not in kwargs: type_from_args = _type_from_args(args) if isinstance(type_from_args, sqltypes.ARRAY): kwargs['type_'] = type_from_args else: - kwargs['type_'] = sqltypes.ARRAY(type_from_args) + kwargs['type_'] = default_array_type(type_from_args) kwargs['_parsed_args'] = args super(array_agg, self).__init__(*args, **kwargs) diff --git a/test/dialect/postgresql/test_compiler.py b/test/dialect/postgresql/test_compiler.py index 66764fcc21..5c4d72c670 100644 --- a/test/dialect/postgresql/test_compiler.py +++ b/test/dialect/postgresql/test_compiler.py @@ -7,6 +7,7 @@ from sqlalchemy import testing from sqlalchemy import Sequence, Table, Column, Integer, update, String,\ func, MetaData, Enum, Index, and_, delete, select, cast, text, \ Text, null +from sqlalchemy import types as sqltypes from sqlalchemy.dialects.postgresql import ExcludeConstraint, array from sqlalchemy import exc, schema from sqlalchemy.dialects import postgresql @@ -16,6 +17,8 @@ from sqlalchemy.sql import table, column, operators, literal_column from sqlalchemy.sql import util as sql_util from sqlalchemy.util import u, OrderedDict from sqlalchemy.dialects.postgresql import aggregate_order_by, insert +from sqlalchemy.dialects.postgresql import array_agg as pg_array_agg +from sqlalchemy.dialects.postgresql import ARRAY as PG_ARRAY class SequenceTest(fixtures.TestBase, AssertsCompiledSQL): @@ -1097,6 +1100,40 @@ class CompileTest(fixtures.TestBase, AssertsCompiledSQL): "AS string_agg_1 FROM table1" ) + def test_pg_array_agg_implicit_pg_array(self): + + expr = pg_array_agg(column('data', Integer)) + assert isinstance(expr.type, PG_ARRAY) + is_(expr.type.item_type._type_affinity, Integer) + + def test_pg_array_agg_uses_base_array(self): + + expr = pg_array_agg(column('data', sqltypes.ARRAY(Integer))) + assert isinstance(expr.type, sqltypes.ARRAY) + assert not isinstance(expr.type, PG_ARRAY) + is_(expr.type.item_type._type_affinity, Integer) + + def test_pg_array_agg_uses_pg_array(self): + + expr = pg_array_agg(column('data', PG_ARRAY(Integer))) + assert isinstance(expr.type, PG_ARRAY) + is_(expr.type.item_type._type_affinity, Integer) + + def test_pg_array_agg_explicit_base_array(self): + + expr = pg_array_agg(column( + 'data', sqltypes.ARRAY(Integer)), type_=sqltypes.ARRAY(Integer)) + assert isinstance(expr.type, sqltypes.ARRAY) + assert not isinstance(expr.type, PG_ARRAY) + is_(expr.type.item_type._type_affinity, Integer) + + def test_pg_array_agg_explicit_pg_array(self): + + expr = pg_array_agg(column( + 'data', sqltypes.ARRAY(Integer)), type_=PG_ARRAY(Integer)) + assert isinstance(expr.type, PG_ARRAY) + is_(expr.type.item_type._type_affinity, Integer) + def test_aggregate_order_by_adapt(self): m = MetaData() table = Table('table1', m, Column('a', Integer), Column('b', Integer))