- Feature Name: SQL savepoints
- Status: draft
- Start Date: 2019-10-14
- Authors: andrei knz
- RFC PR: #41569
- Cockroach Issue: #10735 #28842
Remember, you can submit a PR with your RFC before the text is complete. Refer to the README for details.
Table of contents:
- Summary
- Motivation
- Guide-level explanation
- Reference-level explanation
- Drawbacks
- Rationale and Alternatives
- Unresolved questions
This RFC proposes to implement SQL savepoints as supported by PostgreSQL.
Savepoints enable a client to partially roll back a transaction.
This is a feature that is often requested by users, and used prominently in the test suites of 3rd party tools, in particular ORMs. We want to enable these test suites so as to increase our compatibility coverage.
The addition of SQL savepoints is enabled by a recent changes to the KV layer, whereby write intents now preserve the history of sequence numbers that wrote to a key throughout a transaction.
The proposal here also incidentally addresses CockroachDB's own Halloween problem (issue #28842) without additional work.
SQL savepoints are prominently used by 3rd party tools and frameworks. They are key to implement nested transactions, which is a common programming idiom from Java and other object-oriented languages.
A savepoint is a special mark inside a transaction that allows all commands that are executed after it was established to be rolled back, restoring the transaction state to what it was at the time of the savepoint.
See also: https://www.postgresql.org/docs/current/sql-savepoint.html
- Syntax and introduction
- Nesting
- Savepoint name scoping
- Multiple-level commit/rollback
- Behavior in case of errors
- Relationship with client-side retries
-
to establish a savepoint inside a transaction:
SAVEPOINT savepoint_name
The usual PostgreSQL identifier rules apply:
SAVEPOINT foo
andSAVEPOINT Foo
define the same savepoint, whereasSAVEPOINT "Foo"
defines another. -
to roll back a transaction partially to a previously established savepoint:
ROLLBACK TO SAVEPOINT savepoint_name
-
to forget a savepoint, and keep the effects of statements executed after the savepoint was established:
RELEASE SAVEPOINT savepoint_name
For example:
BEGIN;
INSERT INTO table1 VALUES (1);
SAVEPOINT my_savepoint;
INSERT INTO table1 VALUES (2);
ROLLBACK TO SAVEPOINT my_savepoint;
INSERT INTO table1 VALUES (3);
COMMIT;
The above transaction will insert the values 1 and 3, but not 2.
Schema changes and other DDL are supported "under" savepoints and can be partially rolled back without rolling back the entire transaction.
For example:
BEGIN;
CREATE TABLE u(x INT);
SAVEPOINT foo;
CREATE TABLE t(x INT);
INSERT INTO t(x) VALUES (1);
ROLLBACK TO SAVEPOINT foo;
INSERT INTO u(x) VALUES(1);
SAVEPOINT bar;
CREATE TABLE t(x TEXT);
RELEASE SAVEPOINT foo;
INSERT INTO t(x) VALUES ('a');
COMMIT;
This inserts 1 into u and 'a' into t. The table t
with an INT column
does not exist after the transaction commits.
Savepoints can be nested.
For example:
BEGIN;
INSERT INTO table1 VALUES (1);
SAVEPOINT my_savepoint;
INSERT INTO table1 VALUES (2);
SAVEPOINT my_savepoint2;
INSERT INTO table1 VALUES (3);
ROLLBACK TO SAVEPOINT my_savepoint2;
INSERT INTO table1 VALUES (4);
RELEASE my_savepoint;
COMMIT;
This inserts values 1, 2 and 4 but not 3.
Changes partially committed by a savepoint release can be rolled back by an outer savepoint.
For example:
BEGIN;
INSERT INTO table1 VALUES (1);
SAVEPOINT my_savepoint;
INSERT INTO table1 VALUES (2);
SAVEPOINT my_savepoint2;
INSERT INTO table1 VALUES (3);
RELEASE SAVEPOINT my_savepoint2;
ROLLBACK TO SAVEPOINT my_savepoint;
COMMIT;
This inserts only value 1. The value 3 is rolled back alongside 2.
As an extension to the SQL standard, PostgreSQL allows a
SAVEPOINT
statement to shadow an earlier savepoint with the same
name. The name refers to the new savepoint until released/rolled back,
after which the name reverts to referring to the previous savepoint.
For example:
BEGIN;
INSERT INTO table1 VALUES (1);
SAVEPOINT my_savepoint;
INSERT INTO table1 VALUES (2);
SAVEPOINT my_savepoint;
INSERT INTO table1 VALUES (3);
ROLLBACK TO SAVEPOINT my_savepoint;
INSERT INTO table1 VALUES (4);
RELEASE SAVEPOINT my_savepoint;
COMMIT;
This inserts values 1, 2 and 4, but not 3.
RELEASE SAVEPOINT
and ROLLBACK TO SAVEPOINT
can refer to
a a savepoint "higher" in the nesting hierarchy. When this occurs, all
the savepoints "under" the nesting are automatically released/rolled
back too.
For example:
BEGIN;
SAVEPOINT foo;
INSERT INTO table1 VALUES (1);
SAVEPOINT bar;
INSERT INTO table1 VALUES (2);
ROLLBACK TO SAVEPOINT foo;
RELEASE SAVEPOINT bar; -- error: savepoint "bar" does not exist
COMMIT;
BEGIN;
SAVEPOINT foo;
INSERT INTO table1 VALUES (1);
SAVEPOINT bar;
INSERT INTO table1 VALUES (2);
RELEASE SAVEPOINT foo;
ROLLBACK TO SAVEPOINT bar; -- error: savepoint "bar" does not exist
COMMIT;
If an error occurs "under" a savepoint, it is possible to recover an open, "healthy" txn by rolling back the savepoint without rolling back the txn.
For example:
kena=> create table u(x int unique);
CREATE TABLE
kena=> insert into u(x) values(1);
INSERT 0 1
kena=> begin; \
savepoint foo; \
insert into u(x) values (1); \
rollback to savepoint foo; \
insert into u(x) values (2); \
commit;
BEGIN
SAVEPOINT
ERROR: duplicate key value violates unique constraint "u_x_key"
DETAIL: Key (x)=(1) already exists.
ROLLBACK
INSERT 0 1
COMMIT
kena=> select * from u;
x
---
1
2
(2 rows)
(i.e. the second insert succeeds even though the first insert encountered an error)
The mechanism previously used by CockroachDB to drive client-side retries remains valid, with the following restrictions:
- only savepoints with a name starting with the special prefix
cockroach_restart
(includingcockroach_restart
itself but alsocockroach_restart123134
etc) are to be considered a marker for client-side retries. - restart savepoints can only be used at the outmost level of nesting,
i.e.
begin; savepoint cockroach_restart; savepoint foo
is OK, butbegin; savepoint foo; savepoint cockroach_restart
is not.
(more restrictions TBD as informed by implementation)
- Design overview
- Bill of work (preliminary)
- SQL executor changes
- Savepoints and schema changes
- Savepoint rollbacks, MVCC and storage
- TxnCoordSender changes
The overall design can be understood as follows:
-
at the SQL/KV interface, KV operations are associated with sequence numbers (seqnums):
- write operations generate new seqnums, which are stored inside write intents
- the txn object also contains a list of sequence numbers (or ranges thereof) that are rolled back and are to be ignored during reads.
- read operations operate "at" a particular seqnum- a MVCC read that encounters an intent ignores the values written at later seqnums and returns the most recent value at that seqnum instead, also substracting any past value at seqnums marked to be ignored due to rollbacks.
- intent resolution also throws away any values inside the rolled back seqnum ranges.
-
outside of SQL, reads operate at the seqnum of the latest write as is done today; this preserves read-your-own-writes even inside individual KV batches
-
inside SQL, the executor configures the sql/kv interface so that all reads performed on behalf of an "execution step" (usually, a single statement) are performed at the seqnum of the last write in the previous step.
- most statements consist of just one step, whereby the seqnum boundary is in-between statements. However, with CTE mutations, FK checks and cascading updates, there may be multiple steps inside a single complex statement.
- for CPuts specifically, the condition part ignores the configured read seqnum and checks the most recent (non rolled back) write instead. This is needed in mutations.
-
a savepoint is defined by:
- when adding a savepoint, by saving the current write seqnum,
- to roll back a savepoint, by marking the seqnums in-between as rolled back inside the txn object, to be ignored during reads (see above).
- savepoint release is a no-op in kv, and simply updates a name-seqnum mapping inside SQL.
-
conceptually, savepoints define a stack-like structure. Any savepoint has at most one savepoint open "under" it. When releasing or rolling back to a savepoint higher in the stack, all the savepoints in-between are also released or rolled back. In the SQL executor, this is managed using a stack-like data structure.
-
to introduce savepoint semantics in schema updates / DDL, the descriptor caches are extended to store the seqnum alongside the cached descs. Cache entries are not considered/evicted if the seqnum has been rolled back.
Initial analysis, pre-impl, suggests the following Work to be performed:
- SQL execution changes:
- define the new executor
Step()
method and call it where relevant. - integrate with mutation and FK semantics
- connect the current savepoint/rollback logic to the new txncoordsender interface
- define the new executor
- SQL metadata:
- evict entries from desc caches upon savepoint rollback
- Storage changes to support rollbacks:
- #41612
- extend the txn proto with a list of ignored seqnum ranges
- extend the MVCC read logic to skip over ignored seqnum ranges
- extend the intent resolution logic to skip over ignored seqnum ranges
- TxnCoordSender changes:
- extend the TxnCoordSender interface with methods to mark a SQL step, create a savepoint and rewind (= mark seqnum ranges as ignored)
- rework the error handling logic in txncoordsender (and perhaps other layers) to avoid marking the txn object as in-error for non-permanent errors
To support nesting and shadowing, the SQL executor maintains a naming environment: a list of mappings from names to savepoints.
In Go, using []struct{name,sp}
or equivalent.
(We do not use a simple map
because savepoints form a stack.)
The most recent, innermost savepoint information is at the end of the list.
When defining a savepoint, a new entry is appended. When releasing or rolling back a savepoint, the latest mapping for that sp name is removed:
- Trying to release or rollback a savepoint using a name that does not exist in that list results in an error 3B001 "savepoint does not exist".
- When the target of a RELEASE is an entry that's earlier than the latest entry, all the entries from that entry to the last are also popped out and released.
- Similarly, if the target of a ROLLBACK is an earlier entry, all the entries in-between are also removed from the environment and (conceptually) rolled back.
- (TBD:) uses of savepoint syntax with
cockroach_restart
special names first verify that the stack of savepoints is empty. If it is not, a usage/syntax/unimplemented error is produced.
The executor also provides a method Step()
on the execution context
(likely planner
or whatever is most relevant at the time of
implementation):
- this is called every time a SQL execution step is reached:
- at the end of execution of each regular statement
- after each top-level mutation CTE in a multi-mutation statement
- in-between the FK and cascading action phases, if any
- the implementation of
Step()
forwards the call to the RootTxn's TxnCoordSenderStep()
method (new), which in turn saves the current write seqnum as reference seqnum for future reads.
At a high level, mutations read at the seqnum of the previous statement and write at new seqnums. In particular, they don't read their own writes and this solves issue #28842.
This raises a couple "interesting" topics:
- Uniqueness violations
- ON CONFLICT processing
- FK existence checks under a single mutation
- FK cascading actions under a single mutation
- Multiple mutations with CTEs
There are really two cases:
-
we insert/modify a single row, and doing so creating a duplicate of a row that was modified in a previous statement (or sequencing step). This case is simple and transparently handled by "read at seqnum of previous step".
-
we insert/modify the same row two times inside the same mutation statement, or two rows such that they are duplicate according to some unique index. Here the problem is seemingly that the 2nd row update will not see the first.
However, when looking more closely there is no new problem here.
All writes to a unique index go through a KV CPut
on the uniqueness key.
By ensuring that CPuts
read their condition at the current write
seqnum, we can always pick up the latest write and detect duplicates.
(CPut will still skip over ignored / rolled back seqnums like other KV ops. It's only the target read seqnum that's ratcheted up to the present for CPut, in contrast to other mvcc ops that will be blocked by the configured target read seqnum.)
This opens a question of whether we need a variant of CPut which does not do this. TBD. (Initial analysis says no.)
Question arises of what to do when the result of ON CONFLICT processing changes a row in a read-modify-write fashion. For example:
INSERT INTO t(x) VALUES (1), (1) ON CONFLICT(x) DO UPDATE SET x = t.x + excluded.x
-- ^^^^^^^^ notice the dup row
Here conceptually the INSERT suggests that the 2nd ON CONFLICT resolution will observe the row as it was left over by the 1st. This would not work with "read at seqnum of previous statement".
The answer here is from a previous discussion around mutations that observed the following:
-
postgres does not support updating the same row two times in an ON CONFLICT clause.
-
it is desirable to batch / pre-compute the ON CONFLICT result values concurrently with the mutation for performance, and we've already established back in 2018 that the lack of support for updating the same row twice in pg makes this optimization possible.
-
the implementation was updated when bringing this logic under the CBO
From here, it follows that we don't care about "read at seqnum" inconsistencies as the current CBO logic already assumes that it's fine to miss earlier conflict resolutions.
FK existence checks must observe the data values post-mutation. For this we introduce a step boundary between the end of the "run" phase (where results were produced for the client) and the FK existence checks.
This way the reads for FK existence checks can see all the writes by the mutation.
(To be checked)
Postgres uses post-statement triggers to process FK cascading actions and existence checks. Cascading actions that result in mutations to other tables themselves append more triggers to run.
Each subsequent step in this cascade of effects is able to read its own writes (for futher FK checks).
We emulate this in CockroachDB by introducing a step boundary between iterations of the cascading algorithm.
It's possible for a single statement to define multiple mutations for example:
WITH
a AS (INSERT ... RETURNING ...),
b AS (INSERT ... RETURNING ...)
SELECT ...
PostgreSQL does not guarantee that the effect of one mutation is visible to another, or even to the later read-only parts of the statement. In fact it requires that all mutations operate on the same data at the beginning of the statement:
More specifically: https://www.postgresql.org/docs/12/queries-with.html
The sub-statements in WITH are executed concurrently with each other and with the main query. Therefore, when using data-modifying statements in WITH, the order in which the specified updates actually happen is unpredictable. All the statements are executed with the same snapshot (see Chapter 13), so they cannot “see” one another's effects on the target tables. This alleviates the effects of the unpredictability of the actual order of row updates, and means that RETURNING data is the only way to communicate changes between different WITH sub-statements and the main query.
So with the logic proposed so far, all the mutations inside the same statement execute from the same read seqnum.
If there is FK work to be done, the first sequencing step necessary for FK checks (to advance the read seqnum) will only occur after all mutations have completed.
(The observations from Uniqueness violations above apply here as well.)
Background:
- descriptors used in a txn are loaded in a desc cache (TableCollection + db desc cache) that's (currently) invalidated at the end of the txn
- DDL statements work as follows:
- they modify the descriptor by adding "pending mutation" records and persist the descs embedding these records using the current KV txn context
- additional DDL may add more mutations and perform further KV updates
- at the end of a SQL txn, after the KV txn commits, a "schema changer" job/process is kicked off to resolve the mutation entries on descriptors. The mutation records are read using AS OF SYSTEM TIME with the KV commit timestamp and the final desc post-mutations is written back to kv (possibly at a later timestamp).
To implement SQL savepoints:
-
the descriptor caches must be invalidated when a savepoint is rolled back. To achieve this:
- we extend the caches to store the savepoint object (write seqnum) under which a descriptor was loaded (= the most recent savepoint that's still open).
- when releasing a savepoint, we re-annotate all the cached descriptors cached "under" the savepoint(s) being released, to become cached "under" the savepoint that's still open. (Alternatively, we could flush the desc caches entirely. This is simpler but perf impact unknown. TBD.)
- when rolling back a savepoint, we evict entries from the cache that correspond to a seqnum being rolled back.
-
we let the further processing of DDL statements proceed using KV operations as usual. We expect that the rest of the KV/SQL semantics will make schema changes "just work":
- pending mutation records are cancelled/rolled back as any other KV write
- when the KV txn commits, only the KV writes corresponding to non-rolledback savepoints have been persisted, so any DDL changes "under" savepoints become invisible.
- the schema changer task that starts after that will thus only see the remaining (committed/released) mutation records and the DDL semantics should "just work".
(This will be validated by experimentation in a prototype).
Background:
- for a given KV pair, any write by a txn is done via a write intent
- if there is another write inside the txn, we append the new write
to the intent already laid. After two writes, we'd have separately:
- two seqnums
- two values
- two mvcc timestamps
We need to keep both values around even without savepoints because we may reorder KV reads and writes. For example, in the sequence write1-read1-write2, if the read1 operation happens to be processed after write2, we still want it to see only write1 that is logically in its past.
The way this works today is that each KV operation also has a metadata field which tells it "at which seqnum" it should read. The MVCC read logic, when it encounters an intent, scans the entries in the intent from last to first, and ignores all writes performed at a later seqnum.
For savepoints, this needs to be extended as follows:
-
the txn metadata, which is available in the context of every MVCC read, is extended by a new field "list of ignored seqnum ranges".
-
when a MVCC reads finds an intent and scans to find the first value that's no later than the current read seqnum, it should also skip over any value written at a seqnum also in the ignore list.
Savepoint rollbacks are implemented by populating the ignore list with the range of seqnums generated from the point the savepoint was last established, to the point of the rollback.
This storage-specific part of the work is described in this issue: #41612
(DRAFT - currently in flux)
Sequence numbers (seqnums) are currently used to identify individual writes in a KV batch. They satisfy two different requirements:
- they are used to uniquely identify individual writes
- they are used to order reads and writes to enable read-your-own-writes even in the presence of concurrency (pipeline, out of order eval etc).
Today the TxnCoordSender (the component that receives KV request batches from SQL) is responsible for generating seqnums. There is a seqnum generator that increases monotonically (throughout a KV txn) for every new write request in a KV batch.
The counter is primarily stored in the roachpb.TxnCoordMeta.Txn
,
then copied for local generation whenever a client.Txn
object is
instantiated on top of that with an interceptor of type
txnSeqNumAllocator
(this then uses its own internal seqGen
field
until it re-syncs with the TxnCoordMeta
via populateMetaLocked()
.
Each read request is currently configured (by TxnCoordSender) to use the latest generated write seqnum prior.
When an intent is written, the seqnum of the current write is appended to a list of seqnums already stored in the intent.
The list of all seqnums generated is not stored currently.
(DRAFT - currently in flux)
SQL uses objects of type *client.Txn
. Despite the fact client.Txn
is a struct this does not directly implement or contain the stateful
txn object. Instead it contains (to simplify) two fields:
-
a "type" boolean which can be either
RootTxn
orLeafTxn
.RootTxn indicates that this
client.Txn
is responsible for aggregating all txn state and heartbeats. This type is used for the one client.Txn used in the main SQL executor on the gateway.LeafTxn indicates the
client.Txn
can issue read-only requests on behalf of a RootTxn. The LeafTxn can exist on other nodes than the gateway. The execution machinery is responsible to ensure that the metadata stored in the LeafTxn object is propagated to the RootTxn at the end of its lifecycle. -
a reference to an object of type
client.TxnSender
(client.Txn.mu.sender
), which contains/captures the actual txn state and implements the txn management / request execution logic.
(There are two types of TxnSender currently in use,
client.MockTransactionalSender
, used in tests and
kv.TxnCoordSender
, used for prod. We assume TxnCoordSender
in the
remaining of the RFC)
What happens when a LeafTxn exists concurrently with a RootTxn, and some mutations are performed in the RootTxn while LeafTxn also operates:
- initially (before the LeafTxn starts to exist) there is some seqnum
stored in the
TxnCoordMeta.Txn.Sequence
as a result of previous writes. - the LeafTxn gets instantiated. This points to a copy of the
TxnCoordMeta
(via thesender
reference) which will henceforth forever contain the sameSequence
value. - at any point in time after that, read requests from the LeafTxn
will cause the TxnCoordSender to populate the "read at" seqnum
in the batch requests to be the (current, now immutable) value
of the copy of the
Sequence
field. - concurrently, if mutations are executed on the RootTxn,
this will modify its own (original/main) copy of the
Sequence
field, and ensure that any further reads via the RootTxn, or any subsequent LeafTxn forked off the RootTxn, can see those writes.
At the end of the lifecycle of a LeafTxn, in the case where the execution completed without error, it is necessary for the code to "bring back the metadata" from the LeafTxn client.Txn instance back into the RootTxn instance.
This is done in (TxnCoordSender).augmentMetaLocked()
which:
-
combines "commutative" properties that can evolve in a LeafTxn even when it only sends read requests, such as:
- the txn state (aborted) in case the txn discovers during a read that it was bumped/aborted
- the
ObservedTimestamp
(orOrigTimestampWasObserved
) which "locks in" a commit timestamp and prevents internal refreshes
This is done via
txn.Update()
on the RootTxn, to update it from the LeafTxn.From an explation by Andrei, rephrasing: the model here assumes that if a server produces a TxnProto and you already have one in your hands, you should call txn.Update to refresh your own. There's no clear notion of root vs leaf at that level, the model was designed at a time where we thought that it was always fair and OK to fork a TxnProto, do diverging things with the two copies, then have a way to combine them back together afterwards. Although the model was originally designed to send a TxnProto along with a request, and receive an updated TxnProto along with the response, the mechanism has since them be reused/abused to do all other kinds of concurrent patterns, hence the current over-generality of txn.Update().
(knz:) ISTM that txn.Update for updating a RootTxn from a LeafTxn is overkill; only a subset of the possible changes to a txn object can occur in a LeafTxn.
-
also calls
augmentMetaLocked()
for all interceptors set up on theRootTxn
, which includes:-
in the
txnSpanRefresher
, updating the read spans, which are necessary in subsequent statements, in case of txn conflicts detected at that future time, to check whether the past reads are still current. If we did not bring this metadata properly, then the conflict check may not detect that certain reads are stale and incorrectly decide to refresh the txn (bump its commit ts into the future and retry automatically, instead of pushing the error back to the client) -
perhaps excessively, in
txnPipeliner
, to combine the in-flight writes. This is excessive because today LeafTxns are not allowed to write and thus would not generate in-flight writes. -
perhaps excessively, in
txnSeqNumAllocator
, to bump the sequence number generator in the RootTxn to its max with the LeafTxn generator. This is excessive because today LeafTxns are not allowed to write and thus would not increase their seqnum.
-
(DRAFT - currently in flux)
txn = "current" txn
savepoint a - txn = txn.Savepoint(); txnA := txn
savepoint b - txn = txn.Savepoint(); txnB := txn
savepoint c - txn = txn.Savepoint(); txnC := txn
rollback to a
- txn = txn.RewindTo(txnA)
func (txn) RewindTo(origTxn txn) txn {
if txn == origTxn {
return txn
}
txn.parent.invalidRange.append(txn.seqrange)
txn.Close()
return txn.parent.RewindTo(origTxn)
}
// other
rollback to a
txnC.Rollback()
txnB.Rollback()
...
txnA.sender.ResetReadSequence(txnA.prevSeqToken)
func (txn) Rollback() {
txn.parent.invalidRange.append(txn.seqrange)
txn.Close()
}
func (txn) Close() {
txn.usable = false
txn.parent.Update(txn)
// txn.parent.ResetReadSequence(txn.sender.SetReadSequence())
}
func (txn) Release() { txn.Close() }
func (this txn) Savepoint() txn {
prevSeqToken := txn.sender.SetReadSequence()
return txn{
parent: this,
prevSeqToken: prevSeqToken,
sender: this.sender.Clone() -- new txn proto
// (we'd like to share the sender, but because
// the child savepoint can encounter retry errors
// that will update the txnproto, and we want to
// keep the ability to throw these changes away with
// a savepoint rollback, we can't share that txnproto
// with the parent savepoint.)
}
}
Also, in SQL executor, txn.sender.SetReadSequence() is called at the end of every statement.
This introduces a separation between:
-
SQL-driven client.Txns where the TxnCoordSender uses the seqnum set hby SetReadSequence() to populate the read requests,
-
general-purpose client.Txns where the TxnCoordSender uses (as previously) the seqnum of the latest write.
The separation between these two is set initially when the
client.Txn is instantiated originally. This becomes either a separate
interceptor or a flag in txnSeqNumAllocator
that changes which field
is used to initialize the Sequence
field in batch
request headers. (either current write seqnum, or fixed read seqnum set by
SetReadSequence()
)
(DRAFT - raw notes)
Andrei: we want a common txn interface which needs to satisfy multiple purposes:
-
for distsql, to send metadata on a node and have that node oeprate on behalf of that txn, not handle retries, then retrieve the metadata back to the root
-
this is currently what is done with leaftxns
-
for the halloween problem, we need to tie a SQL step to a sequence number for its reads
-
for savepoints, we need a way to roll back a range of sequence numbers.
-
for concurrent reads, we want a way to tie concurrent actors to a particular txn ID and epoch number, and also a seqnum
-
error handling currently occurs in txncoordsender. When it sees a retry error it acts upon it regardless of whether there are concurrent actors.
- if two goroutines use a single txncoordsender, this will break
- if there's a txncoordsender on a separate node, this will break (we currently avoid breakage using the restrictions on leaftxns)
-
the various ways we "lock in" a view on a txn are "handles" We have already one type of handle (root/leaf) it seems that savepoints are a new type of handle. Maybe we can avoid creating new handle types and unify these things.
-
nathan: instead of using separate client txns, we could use a single one (RootTxn)
-
knz, yes but then have TxnCoordSender handle the savepoint rollback mechanism using a proto stack
-
nathan: IMHO stack not necessary, the txncoordsender can implement rollbacks by injecting the ignored seqnums itself
-
andrei/knz: what about errors, how can we roll back a txn refresh, write too old error, or other things that currently invalidate the txn proto?
-
andrei: the main issue is that the txncoordsender can push the txn proto into a state cannot be rolled back
-
nathan: I think there's value in delaying decisions (txncoordsender error handling) eg for follower reads
-
andrei/nathan: except in case where we know for sure that the client has to deal with the retry, in which case we want to push this ASAP to the client (eagerly)
-
nathan/andrei: concern about Ben's change that itnroduced an interceptor that recognizes write too old errors, and forces a refresh. This is different from what we do for pushes, which are only detected/processed at time of KV commit. (Although the SQL code has access to the knowledge of whether a txn is pushed and will eventually refresh)
-
nathan: what's the plan to handle concurrent reads when there are retry errors?
-
andrei: today when a txncoordinator sees a retry error it either
- refresh the txn automatically
- closes itself and lets the SQL executor instantiated a new txncoordsender
-
knz: how is this broken? ISTM root/leaf handle this
-
nathan: three paths. One in the leaf, two in the root.
- in the root, when kv ops are performed by the root itself
- in the root, when importing metadata from leaves
-
the problem is when handling in the root while leaves may still be running. What we don't want is a leaf observing the effects of error resolution (incl intent resolution) befor ethey're done running.
-
architecturally, the sql executor is the entity that knows that leaves are done. The problem is that txncoordinator eagerly processes these errors before it learns from sql executor that the leaves are done.
-
-
andrei see #24798
-
nathan: an example thing that would benefit from deferring eror handling to the end of the stmt. See the update example.
-
andrei: there's use for both. Try to refresh like we do know. If that fails, let the stmt complete (in error) then restart the stmt (if no result was returned to the client)
-
nathan: what's the benefit of doing partial handling inline?
-
andrei: it can swallow errors. Prevents a full query abort.
-
(knz: note that refreshes are disabled, even in a rootxn, at the level of one batch. If any write command in a batch has succeeded, we can't refresh.)
- how to organize the administration of LeafTxns in the presence of savepoints.
(DRAFT - currently in flux)
Currently TxnCoordSender maintains internal state to mark a transaction as "in error". this is marked after any KV error responses, not just cases where the TxnProto was detected as aborted due to a concurrent txn.
For example, TxnCoordSender marks a txn as "in error" after a failed CPut, even though it's conceivable to continue operating the txn regardless of whether CPut succeeds or not.
This needs to be modified to properly restore the "non-error" state in TxnCoordSender when rolling back a savepoint.
Example errors that cause "permanent", non-rewindable error states:
- txn state aborted
- internal errors in the txncoordsender itself
- internal kv errors due to inconsistent metadata struct
- assertion errors of various kinds
Example errors that can be rewinded:
- txn retry errors
- network errors
- replica unavailable errors
- permanent storage errors - i/o etc
- non-permanent storage errors eg duplicate key
-
we want to preserve the ability to run concurrently.
-
concurrency in txns: have the concurrent actors have a handle on a txn that's locked at a particular epoch
This feature introduces more complexity in the SQL executor.
(to be populated)
There are two general design directions:
-
mark certain ranges of sequence numbers as "rolled back" in the transaction record itself, to be ignored during MVCC reads.
Pros: rollbacks are cheaper
Cons: reads are more expensive
Cons: after a rollback the intents are "still there" and create contention with concurrent txns that touch the rolled back intents.
-
proactively iterate through all intents generated for the current txn and remove the rolled back sequence numbers from the generated intents, including removing the intent if the sequence numbers being rolled back are the only ones remaining.
Pros: reads are cheaper
Pros: rolled back intents also roll back the contention
Pros: may simplify the row locking story (although Tobias found out that PostgreSQL is pretty bad on this so we don't need this pro to be at least as good as pg)
Cons: rollbacks are more expensive
Nathan recommends approach 1, sensing that it requires less work.
(none at this time)