mirror of
https://github.com/pingcap/tla-plus.git
synced 2025-01-14 05:40:09 +08:00
move collapserollback optimization to a new separate directory (#24)
This commit is contained in:
parent
36aa05a586
commit
e3714440db
507
CollapseRollbacks/CollapseRollbacks.tla
Normal file
507
CollapseRollbacks/CollapseRollbacks.tla
Normal file
@ -0,0 +1,507 @@
|
|||||||
|
-------------------------- MODULE CollapseRollbacks -------------------------
|
||||||
|
|
||||||
|
EXTENDS Integers, FiniteSets, Sequences, TLC
|
||||||
|
|
||||||
|
\* The set of transaction keys.
|
||||||
|
CONSTANTS KEY
|
||||||
|
ASSUME KEY # {} \* Keys cannot be empty.
|
||||||
|
|
||||||
|
\* The set of clients to execute a transaction.
|
||||||
|
CONSTANTS CLIENT
|
||||||
|
|
||||||
|
\* Primary keys of all clients (transactions).
|
||||||
|
CONSTANTS CLIENT_PRIMARY_KEY
|
||||||
|
ASSUME CLIENT_PRIMARY_KEY \in [CLIENT -> KEY]
|
||||||
|
|
||||||
|
\* $next_ts$ is the timestamp for transaction. It is increased monotonically,
|
||||||
|
\* so every transaction must have a unique start and commit ts.
|
||||||
|
VARIABLES next_ts
|
||||||
|
|
||||||
|
\* $client_state[c]$ is the state of client.
|
||||||
|
VARIABLES client_state
|
||||||
|
|
||||||
|
\* $client_ts[c]$ is a record of [start_ts: ts, commit_ts: ts].
|
||||||
|
VARIABLES client_ts
|
||||||
|
|
||||||
|
\* $client_key[c]$ is a record of [primary: key, secondary: {key},
|
||||||
|
\* pending: {key}]. Hereby, "pending" denotes the keys that are pending for
|
||||||
|
\* prewrite.
|
||||||
|
VARIABLES client_key
|
||||||
|
|
||||||
|
\* $key_data[k]$ is the set of multi-version data of the key.
|
||||||
|
\* Since we don't care about the concrete value of data, a record of
|
||||||
|
\* [ts: start_ts] is sufficient to represent one data version.
|
||||||
|
VARIABLES key_data
|
||||||
|
|
||||||
|
\* $key_lock[k]$ is the set of lock. A lock is of a record of
|
||||||
|
\* [ts: start_ts, primary: lock]. If $primary$ equals to $k$, it is a primary
|
||||||
|
\* lock, otherwise secondary lock.
|
||||||
|
VARIABLES key_lock
|
||||||
|
|
||||||
|
\* $key_write[k]$ is a sequence of committed version of the key.
|
||||||
|
\* A committed version of the key is a record of [ts: ts, type: type,
|
||||||
|
\* start_ts: start_ts]. $type$ can be "write" or "rollback" depending on record
|
||||||
|
\* type. $start_ts$ field only exists if type is "write". For "write" record,
|
||||||
|
\* $ts$ denotes commit_ts; for "rollback" record, $ts$ denotes start_ts.
|
||||||
|
VARIABLES key_write
|
||||||
|
|
||||||
|
\* Two auxiliary variables for verifying snapshot isolation invariant. These
|
||||||
|
\* variables should not appear in a real-world implementation.
|
||||||
|
\*
|
||||||
|
\* $key_last_read_ts[k]$ denotes the last read timestamp for key $k$, this
|
||||||
|
\* should be monotonic.
|
||||||
|
\*
|
||||||
|
\* $key_si[k]$ denotes if the snapshot isolation invariant is preserved for
|
||||||
|
\* key $k$ so far.
|
||||||
|
VARIABLES key_last_read_ts, key_si
|
||||||
|
|
||||||
|
client_vars == <<client_state, client_ts, client_key>>
|
||||||
|
key_vars == <<key_data, key_lock, key_write, key_last_read_ts, key_si>>
|
||||||
|
vars == <<next_ts, client_vars, key_vars>>
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
Range(m) ==
|
||||||
|
{m[i] : i \in DOMAIN m}
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
\* Checks whether there is a lock of key $k$, whose $ts$ is less than or equal to $ts$.
|
||||||
|
hasLockLE(k, ts) ==
|
||||||
|
\E l \in key_lock[k] : l.ts <= ts
|
||||||
|
|
||||||
|
\* Checks whether there is a lock of key $k$ with $ts$.
|
||||||
|
hasLockEQ(k, ts) ==
|
||||||
|
\E l \in key_lock[k] : l.ts = ts
|
||||||
|
|
||||||
|
\* Returns TRUE if a lock can be cleanup up.
|
||||||
|
\* A lock can be cleaned up iff its ts is less than or equal to $ts$.
|
||||||
|
isStaleLock(l, ts) ==
|
||||||
|
l.ts <= ts
|
||||||
|
|
||||||
|
\* Returns TRUE if we have a stale lock for key $k$.
|
||||||
|
hasStaleLock(k, ts) ==
|
||||||
|
\E l \in key_lock[k] : isStaleLock(l, ts)
|
||||||
|
|
||||||
|
\* Returns the writes with start_ts equals to $ts$.
|
||||||
|
findWriteWithStartTS(k, ts) ==
|
||||||
|
{w \in Range(key_write[k]) : (w.type = "write" /\ w.start_ts = ts)}
|
||||||
|
|
||||||
|
\* Returns the writes with commit_ts equals to $ts$.
|
||||||
|
findWriteWithCommitTS(k, ts) ==
|
||||||
|
{w \in Range(key_write[k]) : (w.type = "write" /\ w.ts = ts)}
|
||||||
|
|
||||||
|
\* Returns TRUE if there is a rollback for key $k$ that is not deleted logically at timestamp $ts$.
|
||||||
|
hasRollbackNotDeleted(k, ts) ==
|
||||||
|
{r \in Range(key_write[k]) : (r.type = "rollback" /\ r.is_deleted = FALSE /\ r.ts = ts )} # {}
|
||||||
|
|
||||||
|
\* Returns TRUE if there is a rollback for key $k$ at timestamp $ts$
|
||||||
|
hasRollback(k, ts) ==
|
||||||
|
{r \in Range(key_write[k]) : (r.type = "rollback" /\ r.ts = ts)} # {}
|
||||||
|
|
||||||
|
\* Returns TRUE if the write is rollback with ts equal to $ts$
|
||||||
|
\* and this rollback is marked as deleted.
|
||||||
|
isRollbackWithDeleted(w, ts) ==
|
||||||
|
(w.type = "rollback" /\ w.ts = ts /\ w.is_deleted = TRUE)
|
||||||
|
|
||||||
|
\* Returns TRUE if there is a rollback at timestamp $ts$
|
||||||
|
\* and this rollback is marked as deleted.
|
||||||
|
hasRollbackWithDeleted(ws, ts) ==
|
||||||
|
{r \in Range(ws) : isRollbackWithDeleted(r, ts)} # {}
|
||||||
|
|
||||||
|
\* Updates $key_si$ for key $k$. If a new version of key $k$ is committed with
|
||||||
|
\* $commit_ts$ < last read timestamp, consider the snapshot isolation invariant
|
||||||
|
\* for key $k$ has been violated.
|
||||||
|
checkSnapshotIsolation(k, commit_ts) ==
|
||||||
|
IF key_last_read_ts[k] >= commit_ts
|
||||||
|
THEN
|
||||||
|
key_si' = [key_si EXCEPT ![k] = FALSE]
|
||||||
|
ELSE
|
||||||
|
UNCHANGED <<key_si>>
|
||||||
|
|
||||||
|
\* Returns the writes with ts less than or equal to $ts$ and is not deleted logically
|
||||||
|
findWriteLessEqualTS(kw, ts) ==
|
||||||
|
{w \in Range(kw) : (w.ts <= ts /\ w.is_deleted = FALSE)}
|
||||||
|
|
||||||
|
\* Returns the write whose $ts$ is the max in all writes
|
||||||
|
getWriteWithMaxTS(w) == CHOOSE x \in w : \A y \in w : x.ts >= y.ts
|
||||||
|
|
||||||
|
\* Deletes element from $seq$ logically
|
||||||
|
logicalDeleteElement(seq, e)==
|
||||||
|
[i \in 1..Len(seq)|->IF seq[i] # e THEN seq[i] ELSE [ts |-> seq[i].ts, type |-> seq[i].type, is_deleted |-> TRUE]]
|
||||||
|
|
||||||
|
\* Removes the pre rollback, whose ts is less than or equal to $ts$.
|
||||||
|
collapsePreRollback(w, ts) ==
|
||||||
|
LET
|
||||||
|
writes == findWriteLessEqualTS(w, ts)
|
||||||
|
IN
|
||||||
|
IF writes # {}
|
||||||
|
THEN
|
||||||
|
LET
|
||||||
|
maxTsWrite == getWriteWithMaxTS(writes)
|
||||||
|
IN
|
||||||
|
IF maxTsWrite.type = "rollback"
|
||||||
|
THEN
|
||||||
|
logicalDeleteElement(w, maxTsWrite)
|
||||||
|
ELSE
|
||||||
|
w
|
||||||
|
ELSE
|
||||||
|
w
|
||||||
|
|
||||||
|
\* Writes rollback record to key write.
|
||||||
|
\* If the rollback with the ts equal to $ts$ exists and is marked as deleted,
|
||||||
|
\* which will be overwritten by the new rollback.
|
||||||
|
writeRollback(ws, ts) ==
|
||||||
|
IF hasRollbackWithDeleted(ws, ts)
|
||||||
|
THEN
|
||||||
|
[i \in 1..Len(ws)|->IF isRollbackWithDeleted(ws[i], ts) THEN ws[i] ELSE [ts |-> ws[i].ts, type |-> ws[i].type, is_deleted |-> FALSE]]
|
||||||
|
ELSE
|
||||||
|
Append(collapsePreRollback(ws, ts), [ts |-> ts, type |-> "rollback", is_deleted |-> FALSE])
|
||||||
|
|
||||||
|
\* Cleans up a stale lock and its data.
|
||||||
|
\* If the lock is a secondary lock, and the assoicated primary lock is cleaned
|
||||||
|
\* up, we can clean up the lock and do,
|
||||||
|
\* 1. If the primary key is committed, we must also commit the secondary key.
|
||||||
|
\* 2. Otherwise, clean up the stale data too.
|
||||||
|
cleanupStaleLock(k, ts) ==
|
||||||
|
LET
|
||||||
|
\* Erases the lock by removing data and lock, and write a rollback record.
|
||||||
|
eraseLock(key, l) ==
|
||||||
|
/\ key_data' = [key_data EXCEPT ![key] = @ \ {[ts |-> l.ts]}]
|
||||||
|
/\ key_lock' = [key_lock EXCEPT ![key] = @ \ {l}]
|
||||||
|
/\ key_write' = [key_write EXCEPT ![key] = writeRollback(key_write[key], l.ts)]
|
||||||
|
IN
|
||||||
|
\E l \in key_lock[k] :
|
||||||
|
/\ isStaleLock(l, ts)
|
||||||
|
/\ UNCHANGED <<key_last_read_ts>>
|
||||||
|
/\ \/ /\ l.primary = k \* this is a primary key, always rollback
|
||||||
|
\* because it is not committed.
|
||||||
|
/\ eraseLock(k, l)
|
||||||
|
/\ UNCHANGED <<key_si>>
|
||||||
|
\/ /\ l.primary # k \* this is a secondary key.
|
||||||
|
/\ LET
|
||||||
|
ws == findWriteWithStartTS(l.primary, l.ts)
|
||||||
|
IN
|
||||||
|
IF ws = {}
|
||||||
|
THEN
|
||||||
|
\* the primary key is not committed, clean up the data.
|
||||||
|
\* Note we should always clean up the corresponding primary
|
||||||
|
\* lock first, then this secondary lock.
|
||||||
|
IF hasRollbackNotDeleted(l.primary, l.ts) = FALSE
|
||||||
|
THEN
|
||||||
|
/\ eraseLock(l.primary, l)
|
||||||
|
/\ UNCHANGED <<key_si>>
|
||||||
|
ELSE
|
||||||
|
/\ eraseLock(k, l)
|
||||||
|
/\ UNCHANGED <<key_si>>
|
||||||
|
ELSE
|
||||||
|
\* the primary key is committed, commit the secondary key.
|
||||||
|
\E w \in ws :
|
||||||
|
/\ key_lock' = [key_lock EXCEPT ![k] = @ \ {l}]
|
||||||
|
/\ key_write' = [key_write EXCEPT ![k] = Append(@, w)]
|
||||||
|
/\ checkSnapshotIsolation(k, w.ts)
|
||||||
|
/\ UNCHANGED <<key_data>>
|
||||||
|
|
||||||
|
\* Cleans up a stale lock when the client encounters one.
|
||||||
|
cleanup(c) ==
|
||||||
|
LET
|
||||||
|
start_ts == client_ts[c].start_ts
|
||||||
|
primary == client_key[c].primary
|
||||||
|
secondary == client_key[c].secondary
|
||||||
|
IN
|
||||||
|
\/ /\ hasStaleLock(primary, start_ts)
|
||||||
|
/\ cleanupStaleLock(primary, start_ts)
|
||||||
|
\/ \E k \in secondary :
|
||||||
|
/\ hasStaleLock(k, start_ts)
|
||||||
|
/\ cleanupStaleLock(k, start_ts)
|
||||||
|
|
||||||
|
\* Reads one key if there is no stale lock, and updates last read timestamp.
|
||||||
|
readKey(c) ==
|
||||||
|
LET
|
||||||
|
start_ts == client_ts[c].start_ts
|
||||||
|
primary == client_key[c].primary
|
||||||
|
secondary == client_key[c].secondary
|
||||||
|
IN
|
||||||
|
\E k \in {primary} \union secondary :
|
||||||
|
/\ ~hasStaleLock(k, start_ts)
|
||||||
|
/\ key_last_read_ts[k] < start_ts
|
||||||
|
/\ key_last_read_ts' = [key_last_read_ts EXCEPT ![k] = start_ts]
|
||||||
|
/\ UNCHANGED <<key_data, key_lock, key_write, key_si>>
|
||||||
|
|
||||||
|
\* Returns TRUE if there is no lock for key $k$, and no any newer writes than
|
||||||
|
\* $ts$.
|
||||||
|
canLockKey(k, ts) ==
|
||||||
|
LET
|
||||||
|
writes == {w \in DOMAIN key_write[k] : key_write[k][w].ts >= ts}
|
||||||
|
IN
|
||||||
|
/\ key_lock[k] = {} \* no any lock for the key.
|
||||||
|
/\ writes = {} \* no any newer writes or rollbacks.
|
||||||
|
|
||||||
|
\* Locks the key and places the data.
|
||||||
|
lockKey(k, start_ts, primary) ==
|
||||||
|
/\ key_lock' = [key_lock EXCEPT ![k] = @ \union {[ts |-> start_ts, primary |-> primary]}]
|
||||||
|
/\ key_data' = [key_data EXCEPT ![k] = @ \union {[ts |-> start_ts]}]
|
||||||
|
/\ UNCHANGED <<key_write, key_last_read_ts, key_si>>
|
||||||
|
|
||||||
|
\* Tries to lock primary key first, then the secondary key.
|
||||||
|
lock(c) ==
|
||||||
|
LET
|
||||||
|
start_ts == client_ts[c].start_ts
|
||||||
|
primary == client_key[c].primary
|
||||||
|
pending == client_key[c].pending
|
||||||
|
IN
|
||||||
|
\* Different from normal percolator protocol, which issues a prewrite on
|
||||||
|
\* primary lock first, then secondary locks, hereby we issues prewrites
|
||||||
|
\* on both primary and secondary locks concurrently. Rollback mechanism
|
||||||
|
\* ensures its correctness.
|
||||||
|
\E k \in pending :
|
||||||
|
/\ canLockKey(k, start_ts)
|
||||||
|
/\ lockKey(k, start_ts, primary)
|
||||||
|
/\ client_key' = [client_key EXCEPT ![c].pending = @ \ {k}]
|
||||||
|
/\ UNCHANGED <<client_state, client_ts>>
|
||||||
|
|
||||||
|
\* Commits the primary key.
|
||||||
|
commitPrimary(c) ==
|
||||||
|
LET
|
||||||
|
start_ts == client_ts[c].start_ts
|
||||||
|
commit_ts == client_ts[c].commit_ts
|
||||||
|
primary == client_key[c].primary
|
||||||
|
IN
|
||||||
|
/\ hasLockEQ(primary, start_ts)
|
||||||
|
/\ key_write' = [key_write EXCEPT ![primary] =
|
||||||
|
Append(@, [ts |-> commit_ts,
|
||||||
|
type |-> "write",
|
||||||
|
start_ts |-> start_ts,
|
||||||
|
is_deleted |-> FALSE])]
|
||||||
|
/\ key_lock' = [key_lock EXCEPT ![primary] = @ \ {[ts |-> start_ts,
|
||||||
|
primary |-> primary]}]
|
||||||
|
/\ checkSnapshotIsolation(primary, commit_ts)
|
||||||
|
/\ UNCHANGED <<key_data, key_last_read_ts>>
|
||||||
|
|
||||||
|
\* Assigns $start_ts$ to the transaction.
|
||||||
|
Start(c) ==
|
||||||
|
/\ client_state[c] = "init"
|
||||||
|
/\ next_ts' = next_ts + 1
|
||||||
|
/\ client_state' = [client_state EXCEPT ![c] = "working"]
|
||||||
|
/\ client_ts' = [client_ts EXCEPT ![c].start_ts = next_ts']
|
||||||
|
/\ UNCHANGED <<key_vars, client_key>>
|
||||||
|
|
||||||
|
\* Does either one thing from these following threes.
|
||||||
|
\* 1. Advances to prewrite phase,
|
||||||
|
\* 2. Tries to clean up one stale lock,
|
||||||
|
\* 3. Reads one key if no stale lock.
|
||||||
|
Get(c) ==
|
||||||
|
/\ client_state[c] = "working"
|
||||||
|
/\ \/ /\ client_state' = [client_state EXCEPT ![c] = "prewriting"]
|
||||||
|
/\ UNCHANGED <<next_ts, key_vars, client_ts, client_key>>
|
||||||
|
\/ /\ cleanup(c)
|
||||||
|
/\ UNCHANGED <<next_ts, client_vars>>
|
||||||
|
\/ /\ readKey(c)
|
||||||
|
/\ UNCHANGED <<next_ts, client_vars>>
|
||||||
|
|
||||||
|
\* Enters commit phase if all locks are granted, otherwise tries to lock the
|
||||||
|
\* primary lock and secondary locks.
|
||||||
|
Prewrite(c) ==
|
||||||
|
/\ client_state[c] = "prewriting"
|
||||||
|
/\ IF client_key[c].pending = {}
|
||||||
|
THEN \* all keys have been pre-written
|
||||||
|
/\ next_ts' = next_ts + 1
|
||||||
|
/\ client_state' = [client_state EXCEPT ![c] = "committing"]
|
||||||
|
/\ client_ts' = [client_ts EXCEPT ![c].commit_ts = next_ts']
|
||||||
|
/\ UNCHANGED <<key_vars, client_key>>
|
||||||
|
ELSE
|
||||||
|
/\ lock(c)
|
||||||
|
/\ UNCHANGED <<next_ts>>
|
||||||
|
|
||||||
|
\* If we commit the primary key successfully, we can think the transaction is
|
||||||
|
\* committed.
|
||||||
|
Commit(c) ==
|
||||||
|
/\ client_state[c] = "committing"
|
||||||
|
/\ commitPrimary(c)
|
||||||
|
/\ client_state' = [client_state EXCEPT ![c] = "committed"]
|
||||||
|
/\ UNCHANGED <<next_ts, client_ts, client_key>>
|
||||||
|
|
||||||
|
\* We can choose to abort at any time if not committed. Hereby, the aborted
|
||||||
|
\* state unifies client crash, client abort and transaction failure. The client
|
||||||
|
\* simply halts when aborted, and leaves cleanup to future transaction.
|
||||||
|
Abort(c) ==
|
||||||
|
/\ client_state[c] # "committed"
|
||||||
|
/\ client_state' = [client_state EXCEPT ![c] = "aborted"]
|
||||||
|
/\ UNCHANGED <<next_ts, client_ts, client_key, key_vars>>
|
||||||
|
|
||||||
|
ClientOp(c) ==
|
||||||
|
\/ Start(c)
|
||||||
|
\/ Get(c)
|
||||||
|
\/ Prewrite(c)
|
||||||
|
\/ Commit(c)
|
||||||
|
\/ Abort(c)
|
||||||
|
|
||||||
|
Next == \E c \in CLIENT : ClientOp(c)
|
||||||
|
|
||||||
|
Init ==
|
||||||
|
LET
|
||||||
|
\* Selects a primary key and use the rest for the secondary keys.
|
||||||
|
chooseKey(ks, c) ==
|
||||||
|
LET
|
||||||
|
primary == CLIENT_PRIMARY_KEY[c]
|
||||||
|
IN
|
||||||
|
[primary |-> primary,
|
||||||
|
secondary |-> ks \ {primary},
|
||||||
|
pending |-> ks]
|
||||||
|
IN
|
||||||
|
/\ next_ts = 0
|
||||||
|
/\ client_state = [c \in CLIENT |-> "init"]
|
||||||
|
/\ client_ts = [c \in CLIENT |-> [start_ts |-> 0, commit_ts |-> 0]]
|
||||||
|
/\ client_key = [c \in CLIENT |-> chooseKey(KEY, c)]
|
||||||
|
/\ key_lock = [k \in KEY |-> {}]
|
||||||
|
/\ key_data = [k \in KEY |-> {}]
|
||||||
|
/\ key_write = [k \in KEY |-> <<>>]
|
||||||
|
/\ key_last_read_ts = [k \in KEY |-> 0]
|
||||||
|
/\ key_si = [k \in KEY |-> TRUE]
|
||||||
|
|
||||||
|
PercolatorSpec == Init /\ [][Next]_vars
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
NextTsTypeInv ==
|
||||||
|
next_ts \in Nat
|
||||||
|
|
||||||
|
ClientStateTypeInv ==
|
||||||
|
client_state \in [CLIENT -> {"init", "working", "prewriting",
|
||||||
|
"committing", "committed", "aborted"}]
|
||||||
|
|
||||||
|
ClientTsTypeInv ==
|
||||||
|
client_ts \in [CLIENT -> [start_ts : Nat, commit_ts : Nat]]
|
||||||
|
|
||||||
|
ClientKeyTypeInv ==
|
||||||
|
client_key \in [CLIENT -> [primary : KEY,
|
||||||
|
secondary : SUBSET KEY,
|
||||||
|
pending : SUBSET KEY]]
|
||||||
|
|
||||||
|
KeyDataTypeInv ==
|
||||||
|
key_data \in [KEY -> SUBSET [ts : Nat]]
|
||||||
|
|
||||||
|
KeyLockTypeInv ==
|
||||||
|
key_lock \in [KEY -> SUBSET [ts : Nat, primary : KEY]]
|
||||||
|
|
||||||
|
KeyWriteTypeInv ==
|
||||||
|
key_write \in [KEY -> Seq([ts : Nat, type : {"write"}, start_ts : Nat, is_deleted: BOOLEAN] \union
|
||||||
|
[ts : Nat, type : {"rollback"}, is_deleted: BOOLEAN])
|
||||||
|
]
|
||||||
|
|
||||||
|
KeyLastReadTsTypeInv ==
|
||||||
|
key_last_read_ts \in [KEY -> Nat]
|
||||||
|
|
||||||
|
KeySiTypeInv ==
|
||||||
|
key_si \in [KEY -> BOOLEAN]
|
||||||
|
|
||||||
|
TypeInvariant ==
|
||||||
|
/\ NextTsTypeInv
|
||||||
|
/\ ClientStateTypeInv
|
||||||
|
/\ ClientTsTypeInv
|
||||||
|
/\ ClientKeyTypeInv
|
||||||
|
/\ KeyDataTypeInv
|
||||||
|
/\ KeyLockTypeInv
|
||||||
|
/\ KeyWriteTypeInv
|
||||||
|
/\ KeyLastReadTsTypeInv
|
||||||
|
/\ KeySiTypeInv
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
\* The committed write timestamp of one key must be in order, and no two writes
|
||||||
|
\* can overlap. For each write, the commit_ts should be strictly greater than
|
||||||
|
\* start_ts.
|
||||||
|
WriteConsistency ==
|
||||||
|
/\ \A k \in KEY :
|
||||||
|
\A i, j \in 1..Len(key_write[k]) :
|
||||||
|
(/\ i < j
|
||||||
|
/\ key_write[k][i].type = "write"
|
||||||
|
/\ key_write[k][j].type = "write"
|
||||||
|
) =>
|
||||||
|
key_write[k][i].ts < key_write[k][j].start_ts
|
||||||
|
/\ \A k \in KEY :
|
||||||
|
\A w \in Range(key_write[k]) :
|
||||||
|
w.type = "write" => w.start_ts < w.ts
|
||||||
|
|
||||||
|
LockConsistency ==
|
||||||
|
\* There should be at most one lock for each key.
|
||||||
|
/\ \A k \in KEY :
|
||||||
|
Cardinality(key_lock[k]) <= 1
|
||||||
|
\* When the client finishes prewriting and is ready for commit, if the
|
||||||
|
\* primary lock exists, all secondary locks should exist.
|
||||||
|
/\ \A c \in CLIENT :
|
||||||
|
(/\ client_state[c] = "committing"
|
||||||
|
/\ hasLockEQ(client_key[c].primary, client_ts[c].start_ts)
|
||||||
|
) =>
|
||||||
|
\A k \in client_key[c].secondary :
|
||||||
|
hasLockEQ(k, client_ts[c].start_ts)
|
||||||
|
|
||||||
|
CommittedConsistency ==
|
||||||
|
\A c \in CLIENT :
|
||||||
|
LET
|
||||||
|
start_ts == client_ts[c].start_ts
|
||||||
|
commit_ts == client_ts[c].commit_ts
|
||||||
|
primary == client_key[c].primary
|
||||||
|
secondary == client_key[c].secondary
|
||||||
|
w == [ts |-> commit_ts, type |-> "write", start_ts |-> start_ts, is_deleted |-> FALSE]
|
||||||
|
IN
|
||||||
|
client_state[c] = "committed" =>
|
||||||
|
\* The primary key lock must be cleaned up, and no any older lock.
|
||||||
|
/\ ~hasLockLE(primary, start_ts)
|
||||||
|
/\ findWriteWithCommitTS(primary, commit_ts) = {w}
|
||||||
|
/\ [ts |-> start_ts] \in key_data[primary]
|
||||||
|
/\ \A k \in secondary :
|
||||||
|
\* The secondary key lock can be empty or not.
|
||||||
|
/\ \/ /\ ~hasLockEQ(k, start_ts)
|
||||||
|
/\ findWriteWithCommitTS(k, commit_ts) = {w}
|
||||||
|
/\ ~hasLockLE(k, start_ts - 1)
|
||||||
|
\/ /\ hasLockEQ(k, start_ts)
|
||||||
|
/\ findWriteWithCommitTS(k, commit_ts) = {}
|
||||||
|
/\ (Len(key_write[k]) > 0 =>
|
||||||
|
\* Lock has not been cleaned up, so the committed
|
||||||
|
\* timestamp of last write must be less than lock's
|
||||||
|
\* start_ts.
|
||||||
|
key_write[k][Len(key_write[k])].ts < start_ts)
|
||||||
|
/\ [ts |-> start_ts] \in key_data[k]
|
||||||
|
|
||||||
|
\* If one transaction is aborted, there should be no committed primary key.
|
||||||
|
AbortedConsistency ==
|
||||||
|
\A c \in CLIENT :
|
||||||
|
(/\ client_state[c] = "aborted"
|
||||||
|
/\ client_ts[c].commit_ts # 0
|
||||||
|
) =>
|
||||||
|
findWriteWithCommitTS(client_key[c].primary, client_ts[c].commit_ts) = {}
|
||||||
|
|
||||||
|
\* For each transaction, we cannot have both committed and rolled back keys.
|
||||||
|
RollbackConsistency ==
|
||||||
|
\A c \in CLIENT :
|
||||||
|
LET
|
||||||
|
start_ts == client_ts[c].start_ts
|
||||||
|
hasWriteKey == \E k \in KEY : findWriteWithStartTS(k, start_ts) # {}
|
||||||
|
hasRollbackKey == \E k \in KEY : hasRollback(k, start_ts)
|
||||||
|
IN
|
||||||
|
start_ts > 0 => ~ (hasWriteKey /\ hasRollbackKey)
|
||||||
|
|
||||||
|
\* For each key, each write or rollback record in write column should have a
|
||||||
|
\* unique start_ts.
|
||||||
|
UniqueWrite ==
|
||||||
|
LET
|
||||||
|
getStartTs(w) ==
|
||||||
|
IF w.type = "write"
|
||||||
|
THEN w.start_ts
|
||||||
|
ELSE w.ts
|
||||||
|
IN
|
||||||
|
\A k \in KEY :
|
||||||
|
Cardinality({getStartTs(w) : w \in Range(key_write[k])}) = Len(key_write[k])
|
||||||
|
|
||||||
|
\* Snapshot isolation invariant should be preserved.
|
||||||
|
SnapshotIsolation ==
|
||||||
|
\A k \in KEY :
|
||||||
|
key_si[k] = TRUE
|
||||||
|
|
||||||
|
--------------------------------------------------------------------------------
|
||||||
|
THEOREM Safety ==
|
||||||
|
PercolatorSpec => [](/\ TypeInvariant
|
||||||
|
/\ WriteConsistency
|
||||||
|
/\ LockConsistency
|
||||||
|
/\ CommittedConsistency
|
||||||
|
/\ AbortedConsistency
|
||||||
|
/\ SnapshotIsolation)
|
||||||
|
|
||||||
|
================================================================================
|
@ -0,0 +1,56 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
|
||||||
|
<launchConfiguration type="org.lamport.tla.toolbox.tool.tlc.modelCheck">
|
||||||
|
<stringAttribute key="TLCCmdLineParameters" value=""/>
|
||||||
|
<stringAttribute key="configurationName" value="Test1"/>
|
||||||
|
<booleanAttribute key="deferLiveness" value="false"/>
|
||||||
|
<intAttribute key="dfidDepth" value="100"/>
|
||||||
|
<booleanAttribute key="dfidMode" value="false"/>
|
||||||
|
<intAttribute key="distributedFPSetCount" value="0"/>
|
||||||
|
<stringAttribute key="distributedNetworkInterface" value="192.168.198.226"/>
|
||||||
|
<intAttribute key="distributedNodesCount" value="1"/>
|
||||||
|
<stringAttribute key="distributedTLC" value="off"/>
|
||||||
|
<stringAttribute key="distributedTLCVMArgs" value=""/>
|
||||||
|
<intAttribute key="fpBits" value="1"/>
|
||||||
|
<intAttribute key="fpIndex" value="1"/>
|
||||||
|
<intAttribute key="maxHeapSize" value="25"/>
|
||||||
|
<intAttribute key="maxSetSize" value="1000000"/>
|
||||||
|
<booleanAttribute key="mcMode" value="true"/>
|
||||||
|
<stringAttribute key="modelBehaviorInit" value=""/>
|
||||||
|
<stringAttribute key="modelBehaviorNext" value=""/>
|
||||||
|
<stringAttribute key="modelBehaviorSpec" value="PercolatorSpec"/>
|
||||||
|
<intAttribute key="modelBehaviorSpecType" value="1"/>
|
||||||
|
<stringAttribute key="modelBehaviorVars" value="key_data, key_lock, next_ts, client_state, key_last_read_ts, client_key, client_ts, key_si, key_write"/>
|
||||||
|
<stringAttribute key="modelComments" value=""/>
|
||||||
|
<booleanAttribute key="modelCorrectnessCheckDeadlock" value="false"/>
|
||||||
|
<listAttribute key="modelCorrectnessInvariants">
|
||||||
|
<listEntry value="1TypeInvariant"/>
|
||||||
|
<listEntry value="1WriteConsistency"/>
|
||||||
|
<listEntry value="1LockConsistency"/>
|
||||||
|
<listEntry value="1CommittedConsistency"/>
|
||||||
|
<listEntry value="1AbortedConsistency"/>
|
||||||
|
<listEntry value="1RollbackConsistency"/>
|
||||||
|
<listEntry value="1UniqueWrite"/>
|
||||||
|
<listEntry value="1SnapshotIsolation"/>
|
||||||
|
</listAttribute>
|
||||||
|
<listAttribute key="modelCorrectnessProperties"/>
|
||||||
|
<stringAttribute key="modelExpressionEval" value=""/>
|
||||||
|
<stringAttribute key="modelParameterActionConstraint" value=""/>
|
||||||
|
<listAttribute key="modelParameterConstants">
|
||||||
|
<listEntry value="KEY;;{1, 2, 3};0;0"/>
|
||||||
|
<listEntry value="CLIENT_PRIMARY_KEY;;c1 :> 1 @@ c2 :> 1 @@ c3 :> 1;0;0"/>
|
||||||
|
<listEntry value="CLIENT;;{c1, c2, c3};1;1"/>
|
||||||
|
</listAttribute>
|
||||||
|
<stringAttribute key="modelParameterContraint" value=""/>
|
||||||
|
<listAttribute key="modelParameterDefinitions"/>
|
||||||
|
<stringAttribute key="modelParameterModelValues" value="{}"/>
|
||||||
|
<stringAttribute key="modelParameterNewDefinitions" value=""/>
|
||||||
|
<intAttribute key="numberOfWorkers" value="4"/>
|
||||||
|
<booleanAttribute key="recover" value="false"/>
|
||||||
|
<stringAttribute key="result.mail.address" value=""/>
|
||||||
|
<intAttribute key="simuAril" value="-1"/>
|
||||||
|
<intAttribute key="simuDepth" value="100"/>
|
||||||
|
<intAttribute key="simuSeed" value="-1"/>
|
||||||
|
<stringAttribute key="specName" value="CollapseRollbacks"/>
|
||||||
|
<stringAttribute key="view" value=""/>
|
||||||
|
<booleanAttribute key="visualizeStateGraph" value="false"/>
|
||||||
|
</launchConfiguration>
|
@ -0,0 +1,56 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
|
||||||
|
<launchConfiguration type="org.lamport.tla.toolbox.tool.tlc.modelCheck">
|
||||||
|
<stringAttribute key="TLCCmdLineParameters" value=""/>
|
||||||
|
<stringAttribute key="configurationName" value="Test2"/>
|
||||||
|
<booleanAttribute key="deferLiveness" value="false"/>
|
||||||
|
<intAttribute key="dfidDepth" value="100"/>
|
||||||
|
<booleanAttribute key="dfidMode" value="false"/>
|
||||||
|
<intAttribute key="distributedFPSetCount" value="0"/>
|
||||||
|
<stringAttribute key="distributedNetworkInterface" value="192.168.198.226"/>
|
||||||
|
<intAttribute key="distributedNodesCount" value="1"/>
|
||||||
|
<stringAttribute key="distributedTLC" value="off"/>
|
||||||
|
<stringAttribute key="distributedTLCVMArgs" value=""/>
|
||||||
|
<intAttribute key="fpBits" value="1"/>
|
||||||
|
<intAttribute key="fpIndex" value="1"/>
|
||||||
|
<intAttribute key="maxHeapSize" value="25"/>
|
||||||
|
<intAttribute key="maxSetSize" value="1000000"/>
|
||||||
|
<booleanAttribute key="mcMode" value="true"/>
|
||||||
|
<stringAttribute key="modelBehaviorInit" value=""/>
|
||||||
|
<stringAttribute key="modelBehaviorNext" value=""/>
|
||||||
|
<stringAttribute key="modelBehaviorSpec" value="PercolatorSpec"/>
|
||||||
|
<intAttribute key="modelBehaviorSpecType" value="1"/>
|
||||||
|
<stringAttribute key="modelBehaviorVars" value="key_data, key_lock, next_ts, client_state, key_last_read_ts, client_key, client_ts, key_si, key_write"/>
|
||||||
|
<stringAttribute key="modelComments" value=""/>
|
||||||
|
<booleanAttribute key="modelCorrectnessCheckDeadlock" value="false"/>
|
||||||
|
<listAttribute key="modelCorrectnessInvariants">
|
||||||
|
<listEntry value="1TypeInvariant"/>
|
||||||
|
<listEntry value="1WriteConsistency"/>
|
||||||
|
<listEntry value="1LockConsistency"/>
|
||||||
|
<listEntry value="1CommittedConsistency"/>
|
||||||
|
<listEntry value="1AbortedConsistency"/>
|
||||||
|
<listEntry value="1RollbackConsistency"/>
|
||||||
|
<listEntry value="1UniqueWrite"/>
|
||||||
|
<listEntry value="1SnapshotIsolation"/>
|
||||||
|
</listAttribute>
|
||||||
|
<listAttribute key="modelCorrectnessProperties"/>
|
||||||
|
<stringAttribute key="modelExpressionEval" value=""/>
|
||||||
|
<stringAttribute key="modelParameterActionConstraint" value=""/>
|
||||||
|
<listAttribute key="modelParameterConstants">
|
||||||
|
<listEntry value="KEY;;{1, 2};0;0"/>
|
||||||
|
<listEntry value="CLIENT_PRIMARY_KEY;;c1 :> 1 @@ c2 :> 2;0;0"/>
|
||||||
|
<listEntry value="CLIENT;;{c1, c2};1;1"/>
|
||||||
|
</listAttribute>
|
||||||
|
<stringAttribute key="modelParameterContraint" value=""/>
|
||||||
|
<listAttribute key="modelParameterDefinitions"/>
|
||||||
|
<stringAttribute key="modelParameterModelValues" value="{}"/>
|
||||||
|
<stringAttribute key="modelParameterNewDefinitions" value=""/>
|
||||||
|
<intAttribute key="numberOfWorkers" value="4"/>
|
||||||
|
<booleanAttribute key="recover" value="false"/>
|
||||||
|
<stringAttribute key="result.mail.address" value=""/>
|
||||||
|
<intAttribute key="simuAril" value="-1"/>
|
||||||
|
<intAttribute key="simuDepth" value="100"/>
|
||||||
|
<intAttribute key="simuSeed" value="-1"/>
|
||||||
|
<stringAttribute key="specName" value="CollapseRollbacks"/>
|
||||||
|
<stringAttribute key="view" value=""/>
|
||||||
|
<booleanAttribute key="visualizeStateGraph" value="false"/>
|
||||||
|
</launchConfiguration>
|
25
CollapseRollbacks/Test1.cfg
Normal file
25
CollapseRollbacks/Test1.cfg
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
\* See Test1.tla.
|
||||||
|
|
||||||
|
CONSTANT
|
||||||
|
c1 = c1
|
||||||
|
c2 = c2
|
||||||
|
c3 = c3
|
||||||
|
KEY <- Key
|
||||||
|
CLIENT <- Client
|
||||||
|
CLIENT_PRIMARY_KEY <- ClientPrimaryKey
|
||||||
|
|
||||||
|
SYMMETRY
|
||||||
|
Symmetry
|
||||||
|
|
||||||
|
SPECIFICATION
|
||||||
|
PercolatorSpec
|
||||||
|
|
||||||
|
INVARIANT
|
||||||
|
TypeInvariant
|
||||||
|
WriteConsistency
|
||||||
|
LockConsistency
|
||||||
|
CommittedConsistency
|
||||||
|
AbortedConsistency
|
||||||
|
RollbackConsistency
|
||||||
|
UniqueWrite
|
||||||
|
SnapshotIsolation
|
16
CollapseRollbacks/Test1.tla
Normal file
16
CollapseRollbacks/Test1.tla
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
--------------------------------- MODULE Test1 ---------------------------------
|
||||||
|
|
||||||
|
EXTENDS CollapseRollbacks, TLC
|
||||||
|
|
||||||
|
\* We use one table with 3 keys and 3 concurrent clients for TLC model checking.
|
||||||
|
\* These 3 clients have the same primary key, so they are considered symmetric.
|
||||||
|
|
||||||
|
CONSTANTS c1, c2, c3
|
||||||
|
|
||||||
|
Key == {1, 2, 3}
|
||||||
|
Client == {c1, c2, c3}
|
||||||
|
ClientPrimaryKey == c1 :> 1 @@ c2 :> 1 @@ c3 :> 1
|
||||||
|
|
||||||
|
Symmetry == Permutations(Client)
|
||||||
|
|
||||||
|
================================================================================
|
21
CollapseRollbacks/Test2.cfg
Normal file
21
CollapseRollbacks/Test2.cfg
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
\* See Test2.tla.
|
||||||
|
|
||||||
|
CONSTANT
|
||||||
|
c1 = c1
|
||||||
|
c2 = c2
|
||||||
|
KEY <- Key
|
||||||
|
CLIENT <- Client
|
||||||
|
CLIENT_PRIMARY_KEY <- ClientPrimaryKey
|
||||||
|
|
||||||
|
SPECIFICATION
|
||||||
|
PercolatorSpec
|
||||||
|
|
||||||
|
INVARIANT
|
||||||
|
TypeInvariant
|
||||||
|
WriteConsistency
|
||||||
|
LockConsistency
|
||||||
|
CommittedConsistency
|
||||||
|
AbortedConsistency
|
||||||
|
RollbackConsistency
|
||||||
|
UniqueWrite
|
||||||
|
SnapshotIsolation
|
14
CollapseRollbacks/Test2.tla
Normal file
14
CollapseRollbacks/Test2.tla
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
--------------------------------- MODULE Test2 ---------------------------------
|
||||||
|
|
||||||
|
EXTENDS CollapseRollbacks, TLC
|
||||||
|
|
||||||
|
\* We use one table with 2 keys and 2 concurrent clients for TLC model checking.
|
||||||
|
\* These 2 clients have primary key 1 and 2 respectively.
|
||||||
|
|
||||||
|
CONSTANTS c1, c2
|
||||||
|
|
||||||
|
Key == {1, 2}
|
||||||
|
Client == {c1, c2}
|
||||||
|
ClientPrimaryKey == c1 :> 1 @@ c2 :> 2
|
||||||
|
|
||||||
|
================================================================================
|
@ -64,7 +64,8 @@ Range(m) ==
|
|||||||
{m[i] : i \in DOMAIN m}
|
{m[i] : i \in DOMAIN m}
|
||||||
|
|
||||||
--------------------------------------------------------------------------------
|
--------------------------------------------------------------------------------
|
||||||
\* Checks whether there is a lock of key $k$, whose $ts$ is less than or equal to $ts$.
|
\* Checks whether there is a lock of key $k$, whose $ts$ is less or equal than
|
||||||
|
\* $ts$.
|
||||||
hasLockLE(k, ts) ==
|
hasLockLE(k, ts) ==
|
||||||
\E l \in key_lock[k] : l.ts <= ts
|
\E l \in key_lock[k] : l.ts <= ts
|
||||||
|
|
||||||
@ -89,24 +90,10 @@ findWriteWithStartTS(k, ts) ==
|
|||||||
findWriteWithCommitTS(k, ts) ==
|
findWriteWithCommitTS(k, ts) ==
|
||||||
{w \in Range(key_write[k]) : (w.type = "write" /\ w.ts = ts)}
|
{w \in Range(key_write[k]) : (w.type = "write" /\ w.ts = ts)}
|
||||||
|
|
||||||
\* Returns TRUE if there is a rollback for key $k$ that is not deleted logically at timestamp $ts$.
|
\* Returns TRUE if there is a rollback for key $k$ at timestamp $ts$.
|
||||||
hasRollbackNotDeleted(k, ts) ==
|
|
||||||
{r \in Range(key_write[k]) : (r.type = "rollback" /\ r.is_deleted = FALSE /\ r.ts = ts )} # {}
|
|
||||||
|
|
||||||
\* Returns TRUE if there is a rollback for key $k$ at timestamp $ts$
|
|
||||||
hasRollback(k, ts) ==
|
hasRollback(k, ts) ==
|
||||||
{r \in Range(key_write[k]) : (r.type = "rollback" /\ r.ts = ts)} # {}
|
{r \in Range(key_write[k]) : (r.type = "rollback" /\ r.ts = ts)} # {}
|
||||||
|
|
||||||
\* Returns TRUE if the write is rollback with ts equal to $ts$
|
|
||||||
\* and this rollback is marked as deleted.
|
|
||||||
isRollbackWithDeleted(w, ts) ==
|
|
||||||
(w.type = "rollback" /\ w.ts = ts /\ w.is_deleted = TRUE)
|
|
||||||
|
|
||||||
\* Returns TRUE if there is a rollback at timestamp $ts$
|
|
||||||
\* and this rollback is marked as deleted.
|
|
||||||
hasRollbackWithDeleted(ws, ts) ==
|
|
||||||
{r \in Range(ws) : isRollbackWithDeleted(r, ts)} # {}
|
|
||||||
|
|
||||||
\* Updates $key_si$ for key $k$. If a new version of key $k$ is committed with
|
\* Updates $key_si$ for key $k$. If a new version of key $k$ is committed with
|
||||||
\* $commit_ts$ < last read timestamp, consider the snapshot isolation invariant
|
\* $commit_ts$ < last read timestamp, consider the snapshot isolation invariant
|
||||||
\* for key $k$ has been violated.
|
\* for key $k$ has been violated.
|
||||||
@ -117,45 +104,6 @@ checkSnapshotIsolation(k, commit_ts) ==
|
|||||||
ELSE
|
ELSE
|
||||||
UNCHANGED <<key_si>>
|
UNCHANGED <<key_si>>
|
||||||
|
|
||||||
\* Returns the writes with ts less than or equal to $ts$ and is not deleted logically
|
|
||||||
findWriteLessEqualTS(kw, ts) ==
|
|
||||||
{w \in Range(kw) : (w.ts <= ts /\ w.is_deleted = FALSE)}
|
|
||||||
|
|
||||||
\* Returns the write whose $ts$ is the max in all writes
|
|
||||||
getWriteWithMaxTS(w) == CHOOSE x \in w : \A y \in w : x.ts >= y.ts
|
|
||||||
|
|
||||||
\* Deletes element from $seq$ logically
|
|
||||||
logicalDeleteElement(seq, e)==
|
|
||||||
[i \in 1..Len(seq)|->IF seq[i] # e THEN seq[i] ELSE [ts |-> seq[i].ts, type |-> seq[i].type, is_deleted |-> TRUE]]
|
|
||||||
|
|
||||||
\* Removes the pre rollback, whose ts is less than or equal to $ts$.
|
|
||||||
collapsePreRollback(w, ts) ==
|
|
||||||
LET
|
|
||||||
writes == findWriteLessEqualTS(w, ts)
|
|
||||||
IN
|
|
||||||
IF writes # {}
|
|
||||||
THEN
|
|
||||||
LET
|
|
||||||
maxTsWrite == getWriteWithMaxTS(writes)
|
|
||||||
IN
|
|
||||||
IF maxTsWrite.type = "rollback"
|
|
||||||
THEN
|
|
||||||
logicalDeleteElement(w, maxTsWrite)
|
|
||||||
ELSE
|
|
||||||
w
|
|
||||||
ELSE
|
|
||||||
w
|
|
||||||
|
|
||||||
\* Writes rollback record to key write.
|
|
||||||
\* If the rollback with the ts equal to $ts$ exists and is marked as deleted,
|
|
||||||
\* which will be overwritten by the new rollback.
|
|
||||||
writeRollback(ws, ts) ==
|
|
||||||
IF hasRollbackWithDeleted(ws, ts)
|
|
||||||
THEN
|
|
||||||
[i \in 1..Len(ws)|->IF isRollbackWithDeleted(ws[i], ts) THEN ws[i] ELSE [ts |-> ws[i].ts, type |-> ws[i].type, is_deleted |-> FALSE]]
|
|
||||||
ELSE
|
|
||||||
Append(collapsePreRollback(ws, ts), [ts |-> ts, type |-> "rollback", is_deleted |-> FALSE])
|
|
||||||
|
|
||||||
\* Cleans up a stale lock and its data.
|
\* Cleans up a stale lock and its data.
|
||||||
\* If the lock is a secondary lock, and the assoicated primary lock is cleaned
|
\* If the lock is a secondary lock, and the assoicated primary lock is cleaned
|
||||||
\* up, we can clean up the lock and do,
|
\* up, we can clean up the lock and do,
|
||||||
@ -167,7 +115,8 @@ cleanupStaleLock(k, ts) ==
|
|||||||
eraseLock(key, l) ==
|
eraseLock(key, l) ==
|
||||||
/\ key_data' = [key_data EXCEPT ![key] = @ \ {[ts |-> l.ts]}]
|
/\ key_data' = [key_data EXCEPT ![key] = @ \ {[ts |-> l.ts]}]
|
||||||
/\ key_lock' = [key_lock EXCEPT ![key] = @ \ {l}]
|
/\ key_lock' = [key_lock EXCEPT ![key] = @ \ {l}]
|
||||||
/\ key_write' = [key_write EXCEPT ![key] = writeRollback(key_write[key], l.ts)]
|
/\ key_write' = [key_write EXCEPT ![key] =
|
||||||
|
Append(@, [ts |-> l.ts, type |-> "rollback"])]
|
||||||
IN
|
IN
|
||||||
\E l \in key_lock[k] :
|
\E l \in key_lock[k] :
|
||||||
/\ isStaleLock(l, ts)
|
/\ isStaleLock(l, ts)
|
||||||
@ -185,7 +134,7 @@ cleanupStaleLock(k, ts) ==
|
|||||||
\* the primary key is not committed, clean up the data.
|
\* the primary key is not committed, clean up the data.
|
||||||
\* Note we should always clean up the corresponding primary
|
\* Note we should always clean up the corresponding primary
|
||||||
\* lock first, then this secondary lock.
|
\* lock first, then this secondary lock.
|
||||||
IF hasRollbackNotDeleted(l.primary, l.ts) = FALSE
|
IF hasRollback(l.primary, l.ts) = FALSE
|
||||||
THEN
|
THEN
|
||||||
/\ eraseLock(l.primary, l)
|
/\ eraseLock(l.primary, l)
|
||||||
/\ UNCHANGED <<key_si>>
|
/\ UNCHANGED <<key_si>>
|
||||||
@ -269,8 +218,7 @@ commitPrimary(c) ==
|
|||||||
/\ key_write' = [key_write EXCEPT ![primary] =
|
/\ key_write' = [key_write EXCEPT ![primary] =
|
||||||
Append(@, [ts |-> commit_ts,
|
Append(@, [ts |-> commit_ts,
|
||||||
type |-> "write",
|
type |-> "write",
|
||||||
start_ts |-> start_ts,
|
start_ts |-> start_ts])]
|
||||||
is_deleted |-> FALSE])]
|
|
||||||
/\ key_lock' = [key_lock EXCEPT ![primary] = @ \ {[ts |-> start_ts,
|
/\ key_lock' = [key_lock EXCEPT ![primary] = @ \ {[ts |-> start_ts,
|
||||||
primary |-> primary]}]
|
primary |-> primary]}]
|
||||||
/\ checkSnapshotIsolation(primary, commit_ts)
|
/\ checkSnapshotIsolation(primary, commit_ts)
|
||||||
@ -382,8 +330,8 @@ KeyLockTypeInv ==
|
|||||||
key_lock \in [KEY -> SUBSET [ts : Nat, primary : KEY]]
|
key_lock \in [KEY -> SUBSET [ts : Nat, primary : KEY]]
|
||||||
|
|
||||||
KeyWriteTypeInv ==
|
KeyWriteTypeInv ==
|
||||||
key_write \in [KEY -> Seq([ts : Nat, type : {"write"}, start_ts : Nat, is_deleted: BOOLEAN] \union
|
key_write \in [KEY -> Seq([ts : Nat, type : {"write"}, start_ts : Nat] \union
|
||||||
[ts : Nat, type : {"rollback"}, is_deleted: BOOLEAN])
|
[ts : Nat, type : {"rollback"}])
|
||||||
]
|
]
|
||||||
|
|
||||||
KeyLastReadTsTypeInv ==
|
KeyLastReadTsTypeInv ==
|
||||||
@ -439,7 +387,7 @@ CommittedConsistency ==
|
|||||||
commit_ts == client_ts[c].commit_ts
|
commit_ts == client_ts[c].commit_ts
|
||||||
primary == client_key[c].primary
|
primary == client_key[c].primary
|
||||||
secondary == client_key[c].secondary
|
secondary == client_key[c].secondary
|
||||||
w == [ts |-> commit_ts, type |-> "write", start_ts |-> start_ts, is_deleted |-> FALSE]
|
w == [ts |-> commit_ts, type |-> "write", start_ts |-> start_ts]
|
||||||
IN
|
IN
|
||||||
client_state[c] = "committed" =>
|
client_state[c] = "committed" =>
|
||||||
\* The primary key lock must be cleaned up, and no any older lock.
|
\* The primary key lock must be cleaned up, and no any older lock.
|
||||||
|
Loading…
Reference in New Issue
Block a user