Commit 749c7c41 authored by Tom Lane's avatar Tom Lane

Fix handling of container types in find_composite_type_dependencies.

find_composite_type_dependencies correctly found columns that are of
the specified type, and columns that are of arrays of that type, but
not columns that are domains or ranges over the given type, its array
type, etc.  The most general way to handle this seems to be to assume
that any type that is directly dependent on the specified type can be
treated as a container type, and processed recursively (allowing us
to handle nested cases such as ranges over domains over arrays ...).
Since a type's array type already has such a dependency, we can drop
the existing special case for the array type.

The very similar logic in get_rels_with_domain was likewise a few
bricks shy of a load, as it supposed that a directly dependent type
could *only* be a sub-domain.  This is already wrong for ranges over
domains, and it'll someday be wrong for arrays over domains.

Add test cases illustrating the problems, and back-patch to all
supported branches.

Discussion: https://postgr.es/m/15268.1502309024@sss.pgh.pa.us
parent a76200de
...@@ -4849,13 +4849,18 @@ ATTypedTableRecursion(List **wqueue, Relation rel, AlterTableCmd *cmd, ...@@ -4849,13 +4849,18 @@ ATTypedTableRecursion(List **wqueue, Relation rel, AlterTableCmd *cmd,
/* /*
* find_composite_type_dependencies * find_composite_type_dependencies
* *
* Check to see if a composite type is being used as a column in some * Check to see if the type "typeOid" is being used as a column in some table
* other table (possibly nested several levels deep in composite types!). * (possibly nested several levels deep in composite types, arrays, etc!).
* Eventually, we'd like to propagate the check or rewrite operation * Eventually, we'd like to propagate the check or rewrite operation
* into other such tables, but for now, just error out if we find any. * into such tables, but for now, just error out if we find any.
* *
* Caller should provide either a table name or a type name (not both) to * Caller should provide either the associated relation of a rowtype,
* report in the error message, if any. * or a type name (not both) for use in the error message, if any.
*
* Note that "typeOid" is not necessarily a composite type; it could also be
* another container type such as an array or range, or a domain over one of
* these things. The name of this function is therefore somewhat historical,
* but it's not worth changing.
* *
* We assume that functions and views depending on the type are not reasons * We assume that functions and views depending on the type are not reasons
* to reject the ALTER. (How safe is this really?) * to reject the ALTER. (How safe is this really?)
...@@ -4868,11 +4873,13 @@ find_composite_type_dependencies(Oid typeOid, Relation origRelation, ...@@ -4868,11 +4873,13 @@ find_composite_type_dependencies(Oid typeOid, Relation origRelation,
ScanKeyData key[2]; ScanKeyData key[2];
SysScanDesc depScan; SysScanDesc depScan;
HeapTuple depTup; HeapTuple depTup;
Oid arrayOid;
/* since this function recurses, it could be driven to stack overflow */
check_stack_depth();
/* /*
* We scan pg_depend to find those things that depend on the rowtype. (We * We scan pg_depend to find those things that depend on the given type.
* assume we can ignore refobjsubid for a rowtype.) * (We assume we can ignore refobjsubid for a type.)
*/ */
depRel = heap_open(DependRelationId, AccessShareLock); depRel = heap_open(DependRelationId, AccessShareLock);
...@@ -4894,8 +4901,22 @@ find_composite_type_dependencies(Oid typeOid, Relation origRelation, ...@@ -4894,8 +4901,22 @@ find_composite_type_dependencies(Oid typeOid, Relation origRelation,
Relation rel; Relation rel;
Form_pg_attribute att; Form_pg_attribute att;
/* Ignore dependees that aren't user columns of relations */ /* Check for directly dependent types */
/* (we assume system columns are never of rowtypes) */ if (pg_depend->classid == TypeRelationId)
{
/*
* This must be an array, domain, or range containing the given
* type, so recursively check for uses of this type. Note that
* any error message will mention the original type not the
* container; this is intentional.
*/
find_composite_type_dependencies(pg_depend->objid,
origRelation, origTypeName);
continue;
}
/* Else, ignore dependees that aren't user columns of relations */
/* (we assume system columns are never of interesting types) */
if (pg_depend->classid != RelationRelationId || if (pg_depend->classid != RelationRelationId ||
pg_depend->objsubid <= 0) pg_depend->objsubid <= 0)
continue; continue;
...@@ -4952,14 +4973,6 @@ find_composite_type_dependencies(Oid typeOid, Relation origRelation, ...@@ -4952,14 +4973,6 @@ find_composite_type_dependencies(Oid typeOid, Relation origRelation,
systable_endscan(depScan); systable_endscan(depScan);
relation_close(depRel, AccessShareLock); relation_close(depRel, AccessShareLock);
/*
* If there's an array type for the rowtype, must check for uses of it,
* too.
*/
arrayOid = get_array_type(typeOid);
if (OidIsValid(arrayOid))
find_composite_type_dependencies(arrayOid, origRelation, origTypeName);
} }
......
...@@ -2787,10 +2787,9 @@ validateDomainConstraint(Oid domainoid, char *ccbin) ...@@ -2787,10 +2787,9 @@ validateDomainConstraint(Oid domainoid, char *ccbin)
* risk by using the weakest suitable lock (ShareLock for most callers). * risk by using the weakest suitable lock (ShareLock for most callers).
* *
* XXX the API for this is not sufficient to support checking domain values * XXX the API for this is not sufficient to support checking domain values
* that are inside composite types or arrays. Currently we just error out * that are inside container types, such as composite types, arrays, or
* if a composite type containing the target domain is stored anywhere. * ranges. Currently we just error out if a container type containing the
* There are not currently arrays of domains; if there were, we could take * target domain is stored anywhere.
* the same approach, but it'd be nicer to fix it properly.
* *
* Generally used for retrieving a list of tests when adding * Generally used for retrieving a list of tests when adding
* new constraints to a domain. * new constraints to a domain.
...@@ -2799,6 +2798,7 @@ static List * ...@@ -2799,6 +2798,7 @@ static List *
get_rels_with_domain(Oid domainOid, LOCKMODE lockmode) get_rels_with_domain(Oid domainOid, LOCKMODE lockmode)
{ {
List *result = NIL; List *result = NIL;
char *domainTypeName = format_type_be(domainOid);
Relation depRel; Relation depRel;
ScanKeyData key[2]; ScanKeyData key[2];
SysScanDesc depScan; SysScanDesc depScan;
...@@ -2806,6 +2806,9 @@ get_rels_with_domain(Oid domainOid, LOCKMODE lockmode) ...@@ -2806,6 +2806,9 @@ get_rels_with_domain(Oid domainOid, LOCKMODE lockmode)
Assert(lockmode != NoLock); Assert(lockmode != NoLock);
/* since this function recurses, it could be driven to stack overflow */
check_stack_depth();
/* /*
* We scan pg_depend to find those things that depend on the domain. (We * We scan pg_depend to find those things that depend on the domain. (We
* assume we can ignore refobjsubid for a domain.) * assume we can ignore refobjsubid for a domain.)
...@@ -2832,20 +2835,32 @@ get_rels_with_domain(Oid domainOid, LOCKMODE lockmode) ...@@ -2832,20 +2835,32 @@ get_rels_with_domain(Oid domainOid, LOCKMODE lockmode)
Form_pg_attribute pg_att; Form_pg_attribute pg_att;
int ptr; int ptr;
/* Check for directly dependent types --- must be domains */ /* Check for directly dependent types */
if (pg_depend->classid == TypeRelationId) if (pg_depend->classid == TypeRelationId)
{ {
Assert(get_typtype(pg_depend->objid) == TYPTYPE_DOMAIN); if (get_typtype(pg_depend->objid) == TYPTYPE_DOMAIN)
{
/* /*
* Recursively add dependent columns to the output list. This is * This is a sub-domain, so recursively add dependent columns
* a bit inefficient since we may fail to combine RelToCheck * to the output list. This is a bit inefficient since we may
* entries when attributes of the same rel have different derived * fail to combine RelToCheck entries when attributes of the
* domain types, but it's probably not worth improving. * same rel have different derived domain types, but it's
* probably not worth improving.
*/ */
result = list_concat(result, result = list_concat(result,
get_rels_with_domain(pg_depend->objid, get_rels_with_domain(pg_depend->objid,
lockmode)); lockmode));
}
else
{
/*
* Otherwise, it is some container type using the domain, so
* fail if there are any columns of this type.
*/
find_composite_type_dependencies(pg_depend->objid,
NULL,
domainTypeName);
}
continue; continue;
} }
...@@ -2882,7 +2897,7 @@ get_rels_with_domain(Oid domainOid, LOCKMODE lockmode) ...@@ -2882,7 +2897,7 @@ get_rels_with_domain(Oid domainOid, LOCKMODE lockmode)
if (OidIsValid(rel->rd_rel->reltype)) if (OidIsValid(rel->rd_rel->reltype))
find_composite_type_dependencies(rel->rd_rel->reltype, find_composite_type_dependencies(rel->rd_rel->reltype,
NULL, NULL,
format_type_be(domainOid)); domainTypeName);
/* /*
* Otherwise, we can ignore relations except those with both * Otherwise, we can ignore relations except those with both
......
...@@ -661,11 +661,28 @@ insert into ddtest2 values(row(-1)); ...@@ -661,11 +661,28 @@ insert into ddtest2 values(row(-1));
alter domain posint add constraint c1 check(value >= 0); alter domain posint add constraint c1 check(value >= 0);
ERROR: cannot alter type "posint" because column "ddtest2.f1" uses it ERROR: cannot alter type "posint" because column "ddtest2.f1" uses it
drop table ddtest2; drop table ddtest2;
-- Likewise for domains within arrays of composite
create table ddtest2(f1 ddtest1[]); create table ddtest2(f1 ddtest1[]);
insert into ddtest2 values('{(-1)}'); insert into ddtest2 values('{(-1)}');
alter domain posint add constraint c1 check(value >= 0); alter domain posint add constraint c1 check(value >= 0);
ERROR: cannot alter type "posint" because column "ddtest2.f1" uses it ERROR: cannot alter type "posint" because column "ddtest2.f1" uses it
drop table ddtest2; drop table ddtest2;
-- Likewise for domains within domains over array of composite
create domain ddtest1d as ddtest1[];
create table ddtest2(f1 ddtest1d);
insert into ddtest2 values('{(-1)}');
alter domain posint add constraint c1 check(value >= 0);
ERROR: cannot alter type "posint" because column "ddtest2.f1" uses it
drop table ddtest2;
drop domain ddtest1d;
-- Doesn't work for ranges, either
create type rposint as range (subtype = posint);
create table ddtest2(f1 rposint);
insert into ddtest2 values('(-1,3]');
alter domain posint add constraint c1 check(value >= 0);
ERROR: cannot alter type "posint" because column "ddtest2.f1" uses it
drop table ddtest2;
drop type rposint;
alter domain posint add constraint c1 check(value >= 0); alter domain posint add constraint c1 check(value >= 0);
create domain posint2 as posint check (value % 2 = 0); create domain posint2 as posint check (value % 2 = 0);
create table ddtest2(f1 posint2); create table ddtest2(f1 posint2);
......
...@@ -451,11 +451,28 @@ insert into ddtest2 values(row(-1)); ...@@ -451,11 +451,28 @@ insert into ddtest2 values(row(-1));
alter domain posint add constraint c1 check(value >= 0); alter domain posint add constraint c1 check(value >= 0);
drop table ddtest2; drop table ddtest2;
-- Likewise for domains within arrays of composite
create table ddtest2(f1 ddtest1[]); create table ddtest2(f1 ddtest1[]);
insert into ddtest2 values('{(-1)}'); insert into ddtest2 values('{(-1)}');
alter domain posint add constraint c1 check(value >= 0); alter domain posint add constraint c1 check(value >= 0);
drop table ddtest2; drop table ddtest2;
-- Likewise for domains within domains over array of composite
create domain ddtest1d as ddtest1[];
create table ddtest2(f1 ddtest1d);
insert into ddtest2 values('{(-1)}');
alter domain posint add constraint c1 check(value >= 0);
drop table ddtest2;
drop domain ddtest1d;
-- Doesn't work for ranges, either
create type rposint as range (subtype = posint);
create table ddtest2(f1 rposint);
insert into ddtest2 values('(-1,3]');
alter domain posint add constraint c1 check(value >= 0);
drop table ddtest2;
drop type rposint;
alter domain posint add constraint c1 check(value >= 0); alter domain posint add constraint c1 check(value >= 0);
create domain posint2 as posint check (value % 2 = 0); create domain posint2 as posint check (value % 2 = 0);
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment