-
Notifications
You must be signed in to change notification settings - Fork 3.9k
Commit
Add a new implementation of `SELECT FOR UPDATE` and `SELECT FOR SHARE` statements. Instead of locking during the initial row fetch, this new implementation constructs a `Lock` operator on the top of the query plan which performs the locking phase using a locking semi-join lookup. During optbuilder we build plans with both `Lock` operators and initial-row-fetch locking. During execbuilder we decide which implementation to use based on the isolation level and whether `optimizer_use_lock_op_for_serializable` is set. If the new implementation is chosen, `Lock` operators become locking semi-LookupJoins. In some cases these new plans will have superfluous lookup joins. A future PR will optimize away some of these superfluous lookup joins. Fixes: #57031, #75457 Epic: CRDB-25322 Release note (sql change): Add a new session variable, `optimizer_use_lock_op_for_serializable`, which when set enables a new implementation of `SELECT FOR UPDATE`. This new implementation of `SELECT FOR UPDATE` acquires row locks *after* any joins and filtering, and always acquires row locks on the primary index of the table being locked. This more closely matches `SELECT FOR UPDATE` behavior in PostgreSQL, but at the cost of more round trips from gateway node to replica leaseholder. Under read committed isolation (and other isolation levels weaker than serializable) we will always use this new implementation of `SELECT FOR UPDATE` regardless of the value of `optimizer_use_lock_op_for_serializable` to ensure correctness.
- Loading branch information
There are no files selected for viewing
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,182 @@ | ||
# LogicTest: !local-mixed-22.2-23.1 | ||
|
||
statement ok | ||
SET CLUSTER SETTING sql.txn.read_committed_syntax.enabled = true | ||
|
||
statement ok | ||
SET SESSION CHARACTERISTICS AS TRANSACTION ISOLATION LEVEL READ COMMITTED | ||
|
||
statement ok | ||
CREATE TABLE abc (a INT PRIMARY KEY, b INT, c INT, INDEX (b), FAMILY (a, b, c)) | ||
|
||
statement ok | ||
INSERT INTO abc VALUES (1, 10, 100), (2, 20, 200), (3, 30, 300) | ||
|
||
statement ok | ||
CREATE TABLE bcd (b INT PRIMARY KEY, c INT, d INT, INDEX (c), FAMILY (b, c, d)) | ||
|
||
statement ok | ||
INSERT INTO bcd VALUES (20, 200, 2000), (30, 300, 3000), (40, 400, 4000) | ||
|
||
statement ok | ||
GRANT ALL on abc TO testuser | ||
|
||
statement ok | ||
GRANT ALL on bcd TO testuser | ||
|
||
# First, hold locks on some rows of abc and bcd. We'll update abc at the end. | ||
|
||
user testuser | ||
|
||
statement ok | ||
BEGIN | ||
|
||
query III rowsort | ||
SELECT * FROM abc WHERE a != 3 FOR UPDATE | ||
---- | ||
1 10 100 | ||
2 20 200 | ||
|
||
query III | ||
SELECT * FROM bcd ORDER BY c DESC LIMIT 2 FOR SHARE | ||
---- | ||
40 400 4000 | ||
30 300 3000 | ||
|
||
# Then ensure we wait on the locks and see the updated rows after commit. | ||
|
||
user root | ||
|
||
# Normal reads do not block. | ||
|
||
query III rowsort | ||
SELECT * FROM abc | ||
---- | ||
1 10 100 | ||
2 20 200 | ||
3 30 300 | ||
|
||
query III rowsort | ||
SELECT * FROM bcd | ||
---- | ||
20 200 2000 | ||
30 300 3000 | ||
40 400 4000 | ||
|
||
# SKIP LOCKED reads do not block. | ||
|
||
query III rowsort | ||
SELECT * FROM abc FOR UPDATE SKIP LOCKED | ||
---- | ||
3 30 300 | ||
|
||
query III rowsort | ||
SELECT * FROM bcd FOR UPDATE SKIP LOCKED | ||
---- | ||
20 200 2000 | ||
30 300 3000 | ||
40 400 4000 | ||
|
||
# Shared reads block on exclusive locks but not on shared locks. | ||
|
||
query III async,rowsort q00 | ||
SELECT * FROM abc FOR SHARE | ||
---- | ||
1 11 101 | ||
2 21 201 | ||
3 30 300 | ||
|
||
query III rowsort | ||
SELECT * FROM bcd FOR SHARE | ||
---- | ||
20 200 2000 | ||
30 300 3000 | ||
40 400 4000 | ||
|
||
# Exclusive reads block on both. | ||
|
||
query III async,rowsort q01 | ||
SELECT * FROM abc FOR UPDATE | ||
---- | ||
1 11 101 | ||
2 21 201 | ||
3 30 300 | ||
|
||
query III async,rowsort q02 | ||
SELECT * FROM bcd FOR UPDATE | ||
---- | ||
20 200 2000 | ||
30 300 3000 | ||
40 400 4000 | ||
|
||
# Try more exclusive-locking queries. | ||
|
||
query I async q03 | ||
SELECT a FROM abc WHERE a = 2 FOR UPDATE | ||
---- | ||
2 | ||
|
||
query I async q04 | ||
SELECT b FROM abc WHERE a = 2 FOR UPDATE | ||
---- | ||
21 | ||
|
||
query I async q05 | ||
SELECT c FROM abc WHERE a = 2 FOR UPDATE | ||
---- | ||
201 | ||
|
||
query I async q06 | ||
SELECT c FROM abc ORDER BY a DESC LIMIT 2 FOR UPDATE | ||
---- | ||
300 | ||
201 | ||
|
||
query I async,rowsort q07 | ||
SELECT a + b + c FROM abc FOR UPDATE | ||
---- | ||
113 | ||
224 | ||
333 | ||
|
||
# Try some joins | ||
|
||
query IIIII async q08 | ||
SELECT * FROM abc JOIN bcd USING (b) FOR SHARE | ||
---- | ||
30 3 300 300 3000 | ||
|
||
query IIIII async q09 | ||
SELECT * FROM abc JOIN bcd USING (c) FOR UPDATE | ||
---- | ||
300 3 30 30 3000 | ||
|
||
user testuser | ||
|
||
statement ok | ||
UPDATE abc SET b = b + 1, c = c + 1 WHERE a != 3 | ||
|
||
statement ok | ||
COMMIT | ||
|
||
user root | ||
|
||
awaitquery q00 | ||
|
||
awaitquery q01 | ||
|
||
awaitquery q02 | ||
|
||
awaitquery q03 | ||
|
||
awaitquery q04 | ||
|
||
awaitquery q05 | ||
|
||
awaitquery q06 | ||
|
||
awaitquery q07 | ||
|
||
awaitquery q08 | ||
|
||
awaitquery q09 |
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.