Skip to content

Commit

Permalink
[#23686] YSQL: Build relcache foreign key list from YB catcache
Browse files Browse the repository at this point in the history
Summary:
Today, when preloading is enabled (either `ysql_catalog_preload_additional_tables` or `ysql_catalog_preload_additional_table_list` is set), we preload the relcache, meaning that we load the relcache entry for every relation in one shot. When we do relcache preloading, we currently don't load foreign key lists—instead, we loading them on-demand. This means on a given connection, the first time each table with a foreign key is referenced, that connection has to fetch the foreign key list from the master leader.

With this revision, we make two changes:
(1) Modify `RelationGetFKeyList` so that it reads foreign keys from the Yugabyte-only `pg_constraint` catcache `YBCONSTRAINTRELIDTYPIDNAME`.  Each time `RelationGetFKeyList` is invoked, it will do a `SearchCatCacheList` on this cache with the partial key `conrelid`.
(2) Whenever we preload the `pg_constraint` cache, we also preload the list cache for the partial key `conrelid`. This has the effect that any calls to `RelationGetFKeyList` after preloading will be able to read from the cached list and will not have to go to master. (A lookup for a table created on a different connection after preloading will still need to go to master).

Both of these changes are controlled by a new GUC `yb_enable_fkey_catcache` (+tserver gflag wrapper) that is on by default.
Jira: DB-12593

Test Plan:
```
./yb_build.sh --cxx-test pg_catalog_perf-test --gtest_filter "*ForeignKeyRelcachePreloadTest*"
```

Reviewers: myang

Reviewed By: myang

Subscribers: yql

Differential Revision: https://phorge.dev.yugabyte.com/D37595
  • Loading branch information
kai-franz committed Sep 27, 2024
1 parent acbb1ba commit 3e93354
Show file tree
Hide file tree
Showing 11 changed files with 292 additions and 11 deletions.
38 changes: 38 additions & 0 deletions java/yb-pgsql/src/test/java/org/yb/pgsql/TestPgForeignKey.java
Original file line number Diff line number Diff line change
Expand Up @@ -311,4 +311,42 @@ public void testInsertConcurrencySerializableIsolation() throws Exception {
testInsertConcurrency(Connection.TRANSACTION_SERIALIZABLE);
}

@Test
public void testAddForeignKeyConcurrent() throws Exception {
try (Statement stmt = connection.createStatement();
Connection ddlConnection = getConnectionBuilder().connect();
Statement ddlStmt = ddlConnection.createStatement()) {
// Create two tables
ddlStmt.execute("CREATE TABLE parent(k int PRIMARY KEY)");
ddlStmt.execute("CREATE TABLE child(k int)");

// Insert data into parent table
stmt.execute("INSERT INTO parent VALUES (1), (2), (3)");

// Start inserting data into child table
stmt.execute("INSERT INTO child VALUES (1)");
stmt.execute("INSERT INTO child VALUES (2)");

// On DDL connection, add foreign key constraint
ddlStmt.execute("ALTER TABLE child ADD CONSTRAINT fk_child_parent " +
"FOREIGN KEY (k) REFERENCES parent(k)");

// Try inserting a valid row on DML connection
stmt.execute("INSERT INTO child VALUES (3)");

// Try inserting an invalid row on DML connection
runInvalidQuery(stmt,
"INSERT INTO child VALUES (4)",
"violates foreign key constraint \"fk_child_parent\"");

// Try inserting an invalid row on the DDL connection
runInvalidQuery(ddlStmt,
"INSERT INTO child VALUES (4)",
"violates foreign key constraint \"fk_child_parent\"");

// Verify the data
assertQuery(stmt, "SELECT k FROM child ORDER BY k",
new Row(1), new Row(2), new Row(3));
}
}
}
38 changes: 27 additions & 11 deletions src/postgres/src/backend/utils/cache/relcache.c
Original file line number Diff line number Diff line change
Expand Up @@ -90,12 +90,14 @@
#include "storage/smgr.h"
#include "utils/array.h"
#include "utils/builtins.h"
#include "utils/catcache.h"
#include "utils/datum.h"
#include "utils/fmgroids.h"
#include "utils/inval.h"
#include "utils/lsyscache.h"
#include "utils/memutils.h"
#include "utils/partcache.h"
#include "utils/relcache.h"
#include "utils/relmapper.h"
#include "utils/resowner_private.h"
#include "utils/snapmgr.h"
Expand Down Expand Up @@ -1936,7 +1938,7 @@ YbCompleteAttrProcessingImpl(const YbAttrProcessorState *state)
{
pfree(constr);
relation->rd_att->constr = NULL;
}
}

/* Fetch rules and triggers that affect this relation */
if (relation->rd_rel->relhasrules)
Expand Down Expand Up @@ -2400,7 +2402,7 @@ YbGetPrefetchableTableInfo(YbPFetchTable table)
[YB_PFETCH_TABLE_PG_CLASS] =
(YbPFetchTableInfo){ RelationRelationId, {YB_TABLE_CACHE_TYPE_CAT_CACHE_WITH_INDEX, .cat_cache = {RELOID, RELNAMENSP}}},
[YB_PFETCH_TABLE_PG_CONSTRAINT] =
(YbPFetchTableInfo){ ConstraintRelationId, {YB_TABLE_CACHE_TYPE_CAT_CACHE_NO_INDEX, .cat_cache = {CONSTROID}}},
(YbPFetchTableInfo){ ConstraintRelationId, {YB_TABLE_CACHE_TYPE_CAT_CACHE_WITH_INDEX, .cat_cache = {CONSTROID, YBCONSTRAINTRELIDTYPIDNAME}}},
[YB_PFETCH_TABLE_PG_DATABASE] =
(YbPFetchTableInfo){ DatabaseRelationId, {YB_TABLE_CACHE_TYPE_CAT_CACHE_NO_INDEX, .cat_cache = {DATABASEOID}}},
[YB_PFETCH_TABLE_PG_DB_ROLE_SETTINGS] =
Expand Down Expand Up @@ -6276,6 +6278,8 @@ RelationGetFKeyList(Relation relation)
HeapTuple htup;
List *oldlist;
MemoryContext oldcxt;
YbCatCListIterator iterator;
bool use_catcache;

/* Quick exit if we already computed the list. */
if (relation->rd_fkeyvalid)
Expand All @@ -6294,17 +6298,24 @@ RelationGetFKeyList(Relation relation)
*/
result = NIL;

/* Prepare to scan pg_constraint for entries having conrelid = this rel. */
ScanKeyInit(&skey,
use_catcache = IsYugaByteEnabled() && yb_enable_fkey_catcache;

if (use_catcache)
{
iterator = YbCatCListIteratorBegin(SearchSysCacheList1(YBCONSTRAINTRELIDTYPIDNAME, RelationGetRelid(relation)));
}
else
{
/* Prepare to scan pg_constraint for entries having conrelid = this rel. */
ScanKeyInit(&skey,
Anum_pg_constraint_conrelid,
BTEqualStrategyNumber, F_OIDEQ,
ObjectIdGetDatum(RelationGetRelid(relation)));

conrel = heap_open(ConstraintRelationId, AccessShareLock);
conscan = systable_beginscan(conrel, ConstraintRelidTypidNameIndexId, true,
conrel = heap_open(ConstraintRelationId, AccessShareLock);
conscan = systable_beginscan(conrel, ConstraintRelidTypidNameIndexId, true,
NULL, 1, &skey);

while (HeapTupleIsValid(htup = systable_getnext(conscan)))
}
while (HeapTupleIsValid(htup = use_catcache ? YbCatCListIteratorGetNext(&iterator) : systable_getnext(conscan)))
{
Form_pg_constraint constraint = (Form_pg_constraint) GETSTRUCT(htup);
ForeignKeyCacheInfo *info;
Expand All @@ -6328,8 +6339,13 @@ RelationGetFKeyList(Relation relation)
result = lappend(result, info);
}

systable_endscan(conscan);
heap_close(conrel, AccessShareLock);
if (use_catcache)
YbCatCListIteratorFree(&iterator);
else
{
systable_endscan(conscan);
heap_close(conrel, AccessShareLock);
}

/* Now save a copy of the completed list in the relcache entry. */
oldcxt = MemoryContextSwitchTo(CacheMemoryContext);
Expand Down
21 changes: 21 additions & 0 deletions src/postgres/src/backend/utils/cache/syscache.c
Original file line number Diff line number Diff line change
Expand Up @@ -1400,6 +1400,26 @@ YbPreloadCatalogCache(int cache_id, int idx_cache_id)
}
break;
}
case CONSTROID:
{
/*
* Add a cache list for YBCONSTRAINTRELIDTYPIDNAME for lookup by conrelid only.
*/
if (!yb_enable_fkey_catcache)
{
is_add_to_list_required = false;
break;
}
if (dest_list)
{
HeapTuple ltp = llast(dest_list);
Form_pg_constraint ltp_struct = (Form_pg_constraint) GETSTRUCT(ltp);
Form_pg_constraint ntp_struct = (Form_pg_constraint) GETSTRUCT(ntp);
if (ntp_struct->conrelid != ltp_struct->conrelid)
dest_list = NIL;
}
break;
}
default:
is_add_to_list_required = false;
break;
Expand Down Expand Up @@ -1433,6 +1453,7 @@ YbPreloadCatalogCache(int cache_id, int idx_cache_id)
switch(cache_id)
{
case PROCOID:
case CONSTROID:
Assert(idx_cache);
dest_cache = idx_cache;
break;
Expand Down
11 changes: 11 additions & 0 deletions src/postgres/src/backend/utils/misc/guc.c
Original file line number Diff line number Diff line change
Expand Up @@ -2652,6 +2652,17 @@ static struct config_bool ConfigureNamesBool[] =
NULL, NULL, NULL
},

{
{"yb_enable_fkey_catcache", PGC_USERSET, DEVELOPER_OPTIONS,
gettext_noop("Enable preloading of foreign key information into the relation cache."),
NULL,
GUC_NOT_IN_SAMPLE
},
&yb_enable_fkey_catcache,
true,
NULL, NULL, NULL
},

/* End-of-list marker */
{
{NULL, 0, 0, NULL, NULL}, NULL, false, NULL, NULL, NULL
Expand Down
1 change: 1 addition & 0 deletions src/postgres/src/backend/utils/misc/pg_yb_utils.c
Original file line number Diff line number Diff line change
Expand Up @@ -1388,6 +1388,7 @@ bool yb_explain_hide_non_deterministic_fields = false;
bool yb_enable_saop_pushdown = true;
int yb_toast_catcache_threshold = -1;
int yb_parallel_range_size = 1024 * 1024;
bool yb_enable_fkey_catcache = true;

YBUpdateOptimizationOptions yb_update_optimization_options = {
.is_enabled = false,
Expand Down
5 changes: 5 additions & 0 deletions src/postgres/src/include/pg_yb_utils.h
Original file line number Diff line number Diff line change
Expand Up @@ -558,6 +558,11 @@ extern int yb_toast_catcache_threshold;
*/
extern int yb_parallel_range_size;

/*
* Enable preloading of foreign key information into the relation cache.
*/
extern bool yb_enable_fkey_catcache;

//------------------------------------------------------------------------------
// GUC variables needed by YB via their YB pointers.
extern int StatementTimeout;
Expand Down
1 change: 1 addition & 0 deletions src/postgres/src/include/utils/guc.h
Original file line number Diff line number Diff line change
Expand Up @@ -269,6 +269,7 @@ extern int yb_explicit_row_locking_batch_size;
extern bool yb_lock_pk_single_rpc;

extern int yb_toast_catcache_threshold;
extern bool yb_enable_fkey_catcache;

extern int temp_file_limit;

Expand Down
75 changes: 75 additions & 0 deletions src/postgres/src/test/regress/expected/yb_dml.out
Original file line number Diff line number Diff line change
Expand Up @@ -18,3 +18,78 @@ EXPLAIN (COSTS OFF) INSERT INTO GH_22967 VALUES ((EXISTS(SELECT 1) in (SELECT tr
(6 rows)

INSERT INTO GH_22967 VALUES ((EXISTS(SELECT 1) in (SELECT true))::INT4), (-10);
-- Test that foreign key constraints are enforced
CREATE TABLE customers (
customer_id SERIAL PRIMARY KEY,
name VARCHAR(100) NOT NULL,
email VARCHAR(100) UNIQUE NOT NULL
);
CREATE TABLE orders (
order_id SERIAL PRIMARY KEY,
order_date DATE NOT NULL,
amount DECIMAL(10, 2) NOT NULL,
customer_id INTEGER NOT NULL,
FOREIGN KEY (customer_id) REFERENCES customers(customer_id)
);
INSERT INTO customers (name, email) VALUES
('Alice Johnson', '[email protected]'),
('Bob Smith', '[email protected]');
INSERT INTO orders (order_date, amount, customer_id) VALUES
('2023-10-01', 250.00, 1),
('2023-10-02', 450.50, 2);
-- Attempt to insert an order with a non-existent customer
INSERT INTO orders (order_date, amount, customer_id) VALUES
('2023-10-03', 300.00, 3);
ERROR: insert or update on table "orders" violates foreign key constraint "orders_customer_id_fkey"
DETAIL: Key (customer_id)=(3) is not present in table "customers".
-- Attempt to delete a customer that still has orders
DELETE FROM customers WHERE customer_id = 2;
ERROR: update or delete on table "customers" violates foreign key constraint "orders_customer_id_fkey" on table "orders"
DETAIL: Key (customer_id)=(2) is still referenced from table "orders".
-- Test cascading deletes
DROP TABLE orders;
CREATE TABLE orders_cascade (
order_id SERIAL PRIMARY KEY,
order_date DATE NOT NULL,
amount DECIMAL(10, 2) NOT NULL,
customer_id INTEGER NOT NULL,
FOREIGN KEY (customer_id) REFERENCES customers(customer_id) ON DELETE CASCADE
);
INSERT INTO orders_cascade (order_date, amount, customer_id) VALUES
('2023-10-01', 250.00, 1),
('2023-10-02', 450.50, 2);
DELETE FROM customers WHERE customer_id = 2;
SELECT * FROM orders_cascade;
order_id | order_date | amount | customer_id
----------+------------+--------+-------------
1 | 10-01-2023 | 250.00 | 1
(1 row)

-- Test adding foreign key constraint using ALTER TABLE ADD CONSTRAINT
CREATE TABLE customers_test (
customer_id SERIAL PRIMARY KEY,
name VARCHAR(100) NOT NULL
);
CREATE TABLE orders_test (
order_id SERIAL PRIMARY KEY,
customer_id INTEGER,
name VARCHAR(100) NOT NULL
);
-- Add foreign key constraint using ALTER TABLE
ALTER TABLE orders_test
ADD CONSTRAINT fk_orders_customers
FOREIGN KEY (customer_id) REFERENCES customers_test(customer_id);
-- Insert valid data
INSERT INTO customers_test (name) VALUES ('Customer 1'), ('Customer 2');
INSERT INTO orders_test (customer_id, name) VALUES (1, 'Order 1'), (2, 'Order 2');
-- Attempt to insert a child with a non-existent customer
INSERT INTO orders_test (customer_id, name) VALUES (3, 'Order 3');
ERROR: insert or update on table "orders_test" violates foreign key constraint "fk_orders_customers"
DETAIL: Key (customer_id)=(3) is not present in table "customers_test".
-- Attempt to delete a customer that still has orders
DELETE FROM customers_test WHERE customer_id = 2;
ERROR: update or delete on table "customers_test" violates foreign key constraint "fk_orders_customers" on table "orders_test"
DETAIL: Key (customer_id)=(2) is still referenced from table "orders_test".
-- Test that invalidation of the foreign key cache works
ALTER TABLE orders_test DROP CONSTRAINT fk_orders_customers;
INSERT INTO orders_test (customer_id, name) VALUES (3, 'Order 3');
78 changes: 78 additions & 0 deletions src/postgres/src/test/regress/sql/yb_dml.sql
Original file line number Diff line number Diff line change
Expand Up @@ -9,3 +9,81 @@
CREATE TABLE GH_22967 (k INT4 PRIMARY KEY);
EXPLAIN (COSTS OFF) INSERT INTO GH_22967 VALUES ((EXISTS(SELECT 1) in (SELECT true))::INT4), (-10);
INSERT INTO GH_22967 VALUES ((EXISTS(SELECT 1) in (SELECT true))::INT4), (-10);


-- Test that foreign key constraints are enforced
CREATE TABLE customers (
customer_id SERIAL PRIMARY KEY,
name VARCHAR(100) NOT NULL,
email VARCHAR(100) UNIQUE NOT NULL
);

CREATE TABLE orders (
order_id SERIAL PRIMARY KEY,
order_date DATE NOT NULL,
amount DECIMAL(10, 2) NOT NULL,
customer_id INTEGER NOT NULL,
FOREIGN KEY (customer_id) REFERENCES customers(customer_id)
);

INSERT INTO customers (name, email) VALUES
('Alice Johnson', '[email protected]'),
('Bob Smith', '[email protected]');

INSERT INTO orders (order_date, amount, customer_id) VALUES
('2023-10-01', 250.00, 1),
('2023-10-02', 450.50, 2);

-- Attempt to insert an order with a non-existent customer
INSERT INTO orders (order_date, amount, customer_id) VALUES
('2023-10-03', 300.00, 3);

-- Attempt to delete a customer that still has orders
DELETE FROM customers WHERE customer_id = 2;

-- Test cascading deletes
DROP TABLE orders;
CREATE TABLE orders_cascade (
order_id SERIAL PRIMARY KEY,
order_date DATE NOT NULL,
amount DECIMAL(10, 2) NOT NULL,
customer_id INTEGER NOT NULL,
FOREIGN KEY (customer_id) REFERENCES customers(customer_id) ON DELETE CASCADE
);
INSERT INTO orders_cascade (order_date, amount, customer_id) VALUES
('2023-10-01', 250.00, 1),
('2023-10-02', 450.50, 2);
DELETE FROM customers WHERE customer_id = 2;
SELECT * FROM orders_cascade;


-- Test adding foreign key constraint using ALTER TABLE ADD CONSTRAINT
CREATE TABLE customers_test (
customer_id SERIAL PRIMARY KEY,
name VARCHAR(100) NOT NULL
);

CREATE TABLE orders_test (
order_id SERIAL PRIMARY KEY,
customer_id INTEGER,
name VARCHAR(100) NOT NULL
);

-- Add foreign key constraint using ALTER TABLE
ALTER TABLE orders_test
ADD CONSTRAINT fk_orders_customers
FOREIGN KEY (customer_id) REFERENCES customers_test(customer_id);

-- Insert valid data
INSERT INTO customers_test (name) VALUES ('Customer 1'), ('Customer 2');
INSERT INTO orders_test (customer_id, name) VALUES (1, 'Order 1'), (2, 'Order 2');

-- Attempt to insert a child with a non-existent customer
INSERT INTO orders_test (customer_id, name) VALUES (3, 'Order 3');

-- Attempt to delete a customer that still has orders
DELETE FROM customers_test WHERE customer_id = 2;

-- Test that invalidation of the foreign key cache works
ALTER TABLE orders_test DROP CONSTRAINT fk_orders_customers;
INSERT INTO orders_test (customer_id, name) VALUES (3, 'Order 3');
Loading

0 comments on commit 3e93354

Please sign in to comment.