Commit 9da86753 authored by Tom Lane's avatar Tom Lane

Reject duplicate column names in foreign key referenced-columns lists.

Such cases are disallowed by the SQL spec, and even if we wanted to allow
them, the semantics seem ambiguous: how should the FK columns be matched up
with the columns of a unique index?  (The matching could be significant in
the presence of opclasses with different notions of equality, so this issue
isn't just academic.)  However, our code did not previously reject such
cases, but instead would either fail to match to any unique index, or
generate a bizarre opclass-lookup error because of sloppy thinking in the
index-matching code.

David Rowley
parent f25e0bf5
...@@ -6738,6 +6738,26 @@ transformFkeyCheckAttrs(Relation pkrel, ...@@ -6738,6 +6738,26 @@ transformFkeyCheckAttrs(Relation pkrel,
bool found_deferrable = false; bool found_deferrable = false;
List *indexoidlist; List *indexoidlist;
ListCell *indexoidscan; ListCell *indexoidscan;
int i,
j;
/*
* Reject duplicate appearances of columns in the referenced-columns list.
* Such a case is forbidden by the SQL standard, and even if we thought it
* useful to allow it, there would be ambiguity about how to match the
* list to unique indexes (in particular, it'd be unclear which index
* opclass goes with which FK column).
*/
for (i = 0; i < numattrs; i++)
{
for (j = i + 1; j < numattrs; j++)
{
if (attnums[i] == attnums[j])
ereport(ERROR,
(errcode(ERRCODE_INVALID_FOREIGN_KEY),
errmsg("foreign key referenced-columns list must not contain duplicates")));
}
}
/* /*
* Get the list of index OIDs for the table from the relcache, and look up * Get the list of index OIDs for the table from the relcache, and look up
...@@ -6750,8 +6770,6 @@ transformFkeyCheckAttrs(Relation pkrel, ...@@ -6750,8 +6770,6 @@ transformFkeyCheckAttrs(Relation pkrel,
{ {
HeapTuple indexTuple; HeapTuple indexTuple;
Form_pg_index indexStruct; Form_pg_index indexStruct;
int i,
j;
indexoid = lfirst_oid(indexoidscan); indexoid = lfirst_oid(indexoidscan);
indexTuple = SearchSysCache1(INDEXRELID, ObjectIdGetDatum(indexoid)); indexTuple = SearchSysCache1(INDEXRELID, ObjectIdGetDatum(indexoid));
...@@ -6770,11 +6788,11 @@ transformFkeyCheckAttrs(Relation pkrel, ...@@ -6770,11 +6788,11 @@ transformFkeyCheckAttrs(Relation pkrel,
heap_attisnull(indexTuple, Anum_pg_index_indpred) && heap_attisnull(indexTuple, Anum_pg_index_indpred) &&
heap_attisnull(indexTuple, Anum_pg_index_indexprs)) heap_attisnull(indexTuple, Anum_pg_index_indexprs))
{ {
/* Must get indclass the hard way */
Datum indclassDatum; Datum indclassDatum;
bool isnull; bool isnull;
oidvector *indclass; oidvector *indclass;
/* Must get indclass the hard way */
indclassDatum = SysCacheGetAttr(INDEXRELID, indexTuple, indclassDatum = SysCacheGetAttr(INDEXRELID, indexTuple,
Anum_pg_index_indclass, &isnull); Anum_pg_index_indclass, &isnull);
Assert(!isnull); Assert(!isnull);
...@@ -6782,7 +6800,13 @@ transformFkeyCheckAttrs(Relation pkrel, ...@@ -6782,7 +6800,13 @@ transformFkeyCheckAttrs(Relation pkrel,
/* /*
* The given attnum list may match the index columns in any order. * The given attnum list may match the index columns in any order.
* Check that each list is a subset of the other. * Check for a match, and extract the appropriate opclasses while
* we're at it.
*
* We know that attnums[] is duplicate-free per the test at the
* start of this function, and we checked above that the number of
* index columns agrees, so if we find a match for each attnums[]
* entry then we must have a one-to-one match in some order.
*/ */
for (i = 0; i < numattrs; i++) for (i = 0; i < numattrs; i++)
{ {
...@@ -6791,6 +6815,7 @@ transformFkeyCheckAttrs(Relation pkrel, ...@@ -6791,6 +6815,7 @@ transformFkeyCheckAttrs(Relation pkrel,
{ {
if (attnums[i] == indexStruct->indkey.values[j]) if (attnums[i] == indexStruct->indkey.values[j])
{ {
opclasses[i] = indclass->values[j];
found = true; found = true;
break; break;
} }
...@@ -6798,24 +6823,6 @@ transformFkeyCheckAttrs(Relation pkrel, ...@@ -6798,24 +6823,6 @@ transformFkeyCheckAttrs(Relation pkrel,
if (!found) if (!found)
break; break;
} }
if (found)
{
for (i = 0; i < numattrs; i++)
{
found = false;
for (j = 0; j < numattrs; j++)
{
if (attnums[j] == indexStruct->indkey.values[i])
{
opclasses[j] = indclass->values[i];
found = true;
break;
}
}
if (!found)
break;
}
}
/* /*
* Refuse to use a deferrable unique/primary key. This is per SQL * Refuse to use a deferrable unique/primary key. This is per SQL
......
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