elog(ERROR, "SPI_connect failed");
/*
- * Prepare a plan for the set default delete operation.
- * Unfortunately we need to do it on every invocation because the
- * default value could potentially change between calls.
+ * Fetch or prepare a saved plan for the set default delete
+ * operation
*/
ri_BuildQueryKey(&qkey, &riinfo, RI_PLAN_SETDEFAULT_DEL_DOUPDATE);
+ if ((qplan = ri_FetchPreparedPlan(&qkey)) == NULL)
{
StringInfoData querybuf;
StringInfoData qualbuf;
}
appendStringInfoString(&querybuf, qualbuf.data);
- /* Prepare the plan, don't save it */
+ /* Prepare and save the plan */
qplan = ri_PlanCheck(querybuf.data, riinfo.nkeys, queryoids,
- &qkey, fk_rel, pk_rel, false);
+ &qkey, fk_rel, pk_rel, true);
}
/*
return PointerGetDatum(NULL);
/*
- * Handle MATCH PARTIAL set null delete.
+ * Handle MATCH PARTIAL set default delete.
*/
case FKCONSTR_MATCH_PARTIAL:
ereport(ERROR,
elog(ERROR, "SPI_connect failed");
/*
- * Prepare a plan for the set default delete operation.
- * Unfortunately we need to do it on every invocation because the
- * default value could potentially change between calls.
+ * Fetch or prepare a saved plan for the set default update
+ * operation
*/
ri_BuildQueryKey(&qkey, &riinfo, RI_PLAN_SETDEFAULT_UPD_DOUPDATE);
+ if ((qplan = ri_FetchPreparedPlan(&qkey)) == NULL)
{
StringInfoData querybuf;
StringInfoData qualbuf;
}
appendStringInfoString(&querybuf, qualbuf.data);
- /* Prepare the plan, don't save it */
+ /* Prepare and save the plan */
qplan = ri_PlanCheck(querybuf.data, riinfo.nkeys, queryoids,
- &qkey, fk_rel, pk_rel, false);
+ &qkey, fk_rel, pk_rel, true);
}
/*
return PointerGetDatum(NULL);
/*
- * Handle MATCH PARTIAL set null delete.
+ * Handle MATCH PARTIAL set default update.
*/
case FKCONSTR_MATCH_PARTIAL:
ereport(ERROR,
(2 rows)
commit;
+--
+-- Test that SET DEFAULT actions recognize updates to default values
+--
+create temp table defp (f1 int primary key);
+NOTICE: CREATE TABLE / PRIMARY KEY will create implicit index "defp_pkey" for table "defp"
+create temp table defc (f1 int default 0
+ references defp on delete set default);
+insert into defp values (0), (1), (2);
+insert into defc values (2);
+select * from defc;
+ f1
+----
+ 2
+(1 row)
+
+delete from defp where f1 = 2;
+select * from defc;
+ f1
+----
+ 0
+(1 row)
+
+delete from defp where f1 = 0; -- fail
+ERROR: update or delete on table "defp" violates foreign key constraint "defc_f1_fkey" on table "defc"
+DETAIL: Key (f1)=(0) is still referenced from table "defc".
+alter table defc alter column f1 set default 1;
+delete from defp where f1 = 0;
+select * from defc;
+ f1
+----
+ 1
+(1 row)
+
+delete from defp where f1 = 1; -- fail
+ERROR: update or delete on table "defp" violates foreign key constraint "defc_f1_fkey" on table "defc"
+DETAIL: Key (f1)=(1) is still referenced from table "defc".
update selfref set a = 456 where a = 123;
select a, b from selfref;
commit;
+
+--
+-- Test that SET DEFAULT actions recognize updates to default values
+--
+create temp table defp (f1 int primary key);
+create temp table defc (f1 int default 0
+ references defp on delete set default);
+insert into defp values (0), (1), (2);
+insert into defc values (2);
+select * from defc;
+delete from defp where f1 = 2;
+select * from defc;
+delete from defp where f1 = 0; -- fail
+alter table defc alter column f1 set default 1;
+delete from defp where f1 = 0;
+select * from defc;
+delete from defp where f1 = 1; -- fail