Commit 20781765 authored by Tom Lane's avatar Tom Lane

Fix incorrect checking of deferred exclusion constraint after a HOT update.

If a row that potentially violates a deferred exclusion constraint is
HOT-updated later in the same transaction, the exclusion constraint would
be reported as violated when the check finally occurs, even if the row(s)
the new row originally conflicted with have since been removed.  This
happened because the wrong TID was passed to check_exclusion_constraint(),
causing the live HOT-updated row to be seen as a conflicting row rather
than recognized as the row-under-test.

Per bug #13148 from Evan Martin.  It's been broken since exclusion
constraints were invented, so back-patch to all supported branches.
parent b4d4ce1d
...@@ -89,9 +89,10 @@ unique_key_recheck(PG_FUNCTION_ARGS) ...@@ -89,9 +89,10 @@ unique_key_recheck(PG_FUNCTION_ARGS)
* because this trigger gets queued only in response to index insertions; * because this trigger gets queued only in response to index insertions;
* which means it does not get queued for HOT updates. The row we are * which means it does not get queued for HOT updates. The row we are
* called for might now be dead, but have a live HOT child, in which case * called for might now be dead, but have a live HOT child, in which case
* we still need to make the check. Therefore we have to use * we still need to make the check --- effectively, we're applying the
* heap_hot_search, not just HeapTupleSatisfiesVisibility as is done in * check against the live child row, although we can use the values from
* the comparable test in RI_FKey_check. * this row since by definition all columns of interest to us are the
* same.
* *
* This might look like just an optimization, because the index AM will * This might look like just an optimization, because the index AM will
* make this identical test before throwing an error. But it's actually * make this identical test before throwing an error. But it's actually
...@@ -159,7 +160,9 @@ unique_key_recheck(PG_FUNCTION_ARGS) ...@@ -159,7 +160,9 @@ unique_key_recheck(PG_FUNCTION_ARGS)
{ {
/* /*
* Note: this is not a real insert; it is a check that the index entry * Note: this is not a real insert; it is a check that the index entry
* that has already been inserted is unique. * that has already been inserted is unique. Passing t_self is
* correct even if t_self is now dead, because that is the TID the
* index will know about.
*/ */
index_insert(indexRel, values, isnull, &(new_row->t_self), index_insert(indexRel, values, isnull, &(new_row->t_self),
trigdata->tg_relation, UNIQUE_CHECK_EXISTING); trigdata->tg_relation, UNIQUE_CHECK_EXISTING);
...@@ -168,10 +171,12 @@ unique_key_recheck(PG_FUNCTION_ARGS) ...@@ -168,10 +171,12 @@ unique_key_recheck(PG_FUNCTION_ARGS)
{ {
/* /*
* For exclusion constraints we just do the normal check, but now it's * For exclusion constraints we just do the normal check, but now it's
* okay to throw error. * okay to throw error. In the HOT-update case, we must use the live
* HOT child's TID here, else check_exclusion_constraint will think
* the child is a conflict.
*/ */
check_exclusion_constraint(trigdata->tg_relation, indexRel, indexInfo, check_exclusion_constraint(trigdata->tg_relation, indexRel, indexInfo,
&(new_row->t_self), values, isnull, &tmptid, values, isnull,
estate, false); estate, false);
} }
......
...@@ -467,6 +467,7 @@ DROP TABLE circles; ...@@ -467,6 +467,7 @@ DROP TABLE circles;
CREATE TABLE deferred_excl ( CREATE TABLE deferred_excl (
f1 int, f1 int,
f2 int,
CONSTRAINT deferred_excl_con EXCLUDE (f1 WITH =) INITIALLY DEFERRED CONSTRAINT deferred_excl_con EXCLUDE (f1 WITH =) INITIALLY DEFERRED
); );
...@@ -482,6 +483,15 @@ INSERT INTO deferred_excl VALUES(3); ...@@ -482,6 +483,15 @@ INSERT INTO deferred_excl VALUES(3);
INSERT INTO deferred_excl VALUES(3); -- no fail here INSERT INTO deferred_excl VALUES(3); -- no fail here
COMMIT; -- should fail here COMMIT; -- should fail here
-- bug #13148: deferred constraint versus HOT update
BEGIN;
INSERT INTO deferred_excl VALUES(2, 1); -- no fail here
DELETE FROM deferred_excl WHERE f1 = 2 AND f2 IS NULL; -- remove old row
UPDATE deferred_excl SET f2 = 2 WHERE f1 = 2;
COMMIT; -- should not fail
SELECT * FROM deferred_excl;
ALTER TABLE deferred_excl DROP CONSTRAINT deferred_excl_con; ALTER TABLE deferred_excl DROP CONSTRAINT deferred_excl_con;
-- This should fail, but worth testing because of HOT updates -- This should fail, but worth testing because of HOT updates
......
...@@ -634,6 +634,7 @@ DROP TABLE circles; ...@@ -634,6 +634,7 @@ DROP TABLE circles;
-- Check deferred exclusion constraint -- Check deferred exclusion constraint
CREATE TABLE deferred_excl ( CREATE TABLE deferred_excl (
f1 int, f1 int,
f2 int,
CONSTRAINT deferred_excl_con EXCLUDE (f1 WITH =) INITIALLY DEFERRED CONSTRAINT deferred_excl_con EXCLUDE (f1 WITH =) INITIALLY DEFERRED
); );
INSERT INTO deferred_excl VALUES(1); INSERT INTO deferred_excl VALUES(1);
...@@ -654,6 +655,19 @@ INSERT INTO deferred_excl VALUES(3); -- no fail here ...@@ -654,6 +655,19 @@ INSERT INTO deferred_excl VALUES(3); -- no fail here
COMMIT; -- should fail here COMMIT; -- should fail here
ERROR: conflicting key value violates exclusion constraint "deferred_excl_con" ERROR: conflicting key value violates exclusion constraint "deferred_excl_con"
DETAIL: Key (f1)=(3) conflicts with existing key (f1)=(3). DETAIL: Key (f1)=(3) conflicts with existing key (f1)=(3).
-- bug #13148: deferred constraint versus HOT update
BEGIN;
INSERT INTO deferred_excl VALUES(2, 1); -- no fail here
DELETE FROM deferred_excl WHERE f1 = 2 AND f2 IS NULL; -- remove old row
UPDATE deferred_excl SET f2 = 2 WHERE f1 = 2;
COMMIT; -- should not fail
SELECT * FROM deferred_excl;
f1 | f2
----+----
1 |
2 | 2
(2 rows)
ALTER TABLE deferred_excl DROP CONSTRAINT deferred_excl_con; ALTER TABLE deferred_excl DROP CONSTRAINT deferred_excl_con;
-- This should fail, but worth testing because of HOT updates -- This should fail, but worth testing because of HOT updates
UPDATE deferred_excl SET f1 = 3; UPDATE deferred_excl SET f1 = 3;
......
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