-
Notifications
You must be signed in to change notification settings - Fork 69
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
RFC: reset TiKV to some certain mvcc snapshot #81
Open
longfangsong
wants to merge
5
commits into
tikv:master
Choose a base branch
from
longfangsong:patch-1
base: master
Could not load branches
Branch not found: {{ refName }}
Loading
Could not load tags
Nothing to show
Loading
Are you sure you want to change the base?
Some commits from the old base branch may be removed from the timeline,
and old review comments may become outdated.
+103
−0
Open
Changes from 2 commits
Commits
Show all changes
5 commits
Select commit
Hold shift + click to select a range
40ac22e
Create 0081-flashback.md
longfangsong 9dc2a77
rename flashback into truncate
longfangsong a2cc55c
Update text/0081-truncate.md
longfangsong 95b955c
add some note about no more write requests should be sent to TiKV dur…
longfangsong a5c7eb9
address comments
longfangsong File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,105 @@ | ||
# RFC: Return TiKV to some certain version | ||
|
||
## Summary | ||
|
||
This proposal introduces a method to return TiKV to some certain version, which could be useful when doing some lossy recovery. | ||
|
||
## Motivation | ||
|
||
The initial motivation is to do lossy recovery on TiKV when deploying TiDB cluster across DCs with DR Auto-sync method. | ||
|
||
When use DR Auto-sync method to deploy TiDB cluster in two DC and | ||
|
||
- there is a network isolation between primary and DR DC | ||
- PD make the replica state of primary DC to be Async after a while | ||
- the primary DC is down | ||
|
||
Then the sync state of the cluster (from the DR DC's PD) is not dependable. In this case, data can be partially sent to the TiKV nodes in DR, and break the ACID consistency. | ||
|
||
Though it is impossible to do lossless recovery in this situation, we can recover the data on TiKV nodes to a certain version (`resolved_ts` in this case), which is ACID consistency. | ||
|
||
## Detailed design | ||
|
||
Basically, we need a method to return TiKV to a certain version (timestamp), we should add a new RPC to kvproto: | ||
longfangsong marked this conversation as resolved.
Show resolved
Hide resolved
|
||
|
||
```protobuf | ||
message TruncateRequest { | ||
longfangsong marked this conversation as resolved.
Show resolved
Hide resolved
|
||
Context context = 1; | ||
uint64 checkpoint_ts = 2; | ||
} | ||
|
||
message TruncateResponse { | ||
string error = 1; | ||
} | ||
|
||
service Tikv { | ||
// … | ||
rpc Truncate(kvrpcpb.TruncateRequest) returns (kvrpcpb.TruncateResponse) {} | ||
longfangsong marked this conversation as resolved.
Show resolved
Hide resolved
|
||
} | ||
``` | ||
|
||
When a TiKV receive `TruncateRequest`, it will create a `TruncateWorker` instance and start the work. | ||
|
||
There are several steps to follow when doing truncate: | ||
|
||
### 1. Wait all committed raft log are applied | ||
|
||
We can just polling each region's `RegionInfo` in a method similar with [`Debugger::region_info`](https://github.com/tikv/tikv/blob/789c99666f2f9faaa6c6e5b021ac0cf7a76ae24e/src/server/debug.rs#L188) does, and wait for `commit_index` to be equal to `apply_index`. | ||
longfangsong marked this conversation as resolved.
Show resolved
Hide resolved
|
||
|
||
### 2. Clean WriteCF and DefaultCF | ||
|
||
Then we can remove all data created by transactions which committed after `checkpoint_ts`, this can be done by remove all `Write` records which `commit_ts > checkpoint_ts` and corresponding data in `DEFAULT_CF` from KVEngine, ie. | ||
|
||
```rust | ||
fn scan_next_batch(&mut self, batch_size: usize) -> Option<Vec<(Vec<u8>, Write)>> { | ||
let mut writes = None; | ||
for _ in 0..batch_size { | ||
if let Some((key, write)) = self.next_write()? { | ||
let commit_ts = Key::decode_ts_from(keys::origin_key(&key)); | ||
let mut writes = writes.get_or_insert(Vec::new()); | ||
if commit_ts > self.ts { | ||
writes.push((key, write)); | ||
} | ||
} else { | ||
return writes; | ||
} | ||
} | ||
writes | ||
} | ||
|
||
pub fn process_next_batch( | ||
&mut self, | ||
batch_size: usize, | ||
wb: &mut RocksWriteBatch, | ||
) -> bool { | ||
longfangsong marked this conversation as resolved.
Show resolved
Hide resolved
|
||
let writes = if let Some(writes) = self.scan_next_batch(batch_size) { | ||
writes | ||
} else { | ||
return false; | ||
} | ||
for (key, write) in writes { | ||
let default_key = Key::append_ts(Key::from_raw(&key), write.start_ts).to_raw().unwrap(); | ||
box_try!(wb.delete_cf(CF_WRITE, &key)); | ||
box_try!(wb.delete_cf(CF_DEFAULT, &default_key)); | ||
} | ||
true | ||
} | ||
``` | ||
|
||
Note we scan and process data in batches, so we can report the progress to the client. | ||
|
||
### 3. Resolve locks once | ||
longfangsong marked this conversation as resolved.
Show resolved
Hide resolved
|
||
|
||
After remove all the writes, we can promise we won't give the user ACID inconsistency data when processing read, however, there might be still locks in lock cf which block some user's operation. | ||
|
||
So we should do resolve locks on all data once. | ||
|
||
This lock resolving process is just like the one used by GC, with safepoint ts equals to current ts. | ||
|
||
## Drawbacks | ||
|
||
This is a **very** unsafe operation, we should prevent the user call this on normal cluster. | ||
|
||
## Alternatives | ||
|
||
- We can also clean the lock cf in step2 instead of do resolve locks. |
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Whose
resolved_ts
? DR DC's or primary DC's?