Skip to content
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

consensus: Revert to last common ancestor #1263

Merged
merged 6 commits into from
Jan 23, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
27 changes: 21 additions & 6 deletions node/src/chain/acceptor.rs
Original file line number Diff line number Diff line change
Expand Up @@ -25,8 +25,9 @@ use crate::chain::header_validation::Validator;

#[allow(dead_code)]
pub(crate) enum RevertTarget {
LastFinalizedState = 0,
LastEpoch = 1,
Commit([u8; 32]),
LastFinalizedState,
LastEpoch,
}

/// Implements block acceptance procedure. This includes block header,
Expand Down Expand Up @@ -407,17 +408,31 @@ impl<DB: database::DB, VM: vm::VMExecution, N: Network> Acceptor<N, DB, VM> {

let target_state_hash = match target {
RevertTarget::LastFinalizedState => {
info!(event = "vm_revert to last finalized state");
let state_hash = self.vm.read().await.revert()?;
let vm = self.vm.read().await;
let state_hash = vm.revert_to_finalized()?;

info!(
event = "vm reverted",
state_root = hex::encode(state_hash)
state_root = hex::encode(state_hash),
is_final = "true",
);

anyhow::Ok(state_hash)
}
_ => unimplemented!(),
RevertTarget::Commit(state_hash) => {
let vm = self.vm.read().await;
let state_hash = vm.revert(state_hash)?;
let is_final = vm.get_finalized_state_root()? == state_hash;

info!(
event = "vm reverted",
state_root = hex::encode(state_hash),
is_final,
);

anyhow::Ok(state_hash)
}
RevertTarget::LastEpoch => unimplemented!(),
}?;

// Delete any block until we reach the target_state_hash, the
Expand Down
30 changes: 25 additions & 5 deletions node/src/chain/fsm.rs
Original file line number Diff line number Diff line change
Expand Up @@ -299,7 +299,7 @@ impl<DB: database::DB, VM: vm::VMExecution, N: Network> InSyncImpl<DB, VM, N> {
// R_B.Iteration < L_B.Iteration
//
// Then we fallback to N_B.PrevBlock and accept N_B
let local_header = acc.db.read().await.view(|t| {
let result = acc.db.read().await.view(|t| {
if let Some((prev_header, _)) =
t.fetch_block_header(&remote_blk.header().prev_block_hash)?
{
Expand All @@ -308,20 +308,23 @@ impl<DB: database::DB, VM: vm::VMExecution, N: Network> InSyncImpl<DB, VM, N> {
if remote_blk.header().iteration
< l_b.header().iteration
{
return Ok(Some(l_b.header().clone()));
return Ok(Some((
l_b.header().clone(),
prev_header.state_hash,
)));
}
}
}

anyhow::Ok(None)
})?;

if let Some(local_header) = local_header {
if let Some((local_header, state_hash)) = result {
match fallback::WithContext::new(acc.deref())
.try_revert(
&local_header,
remote_blk.header(),
RevertTarget::LastFinalizedState,
RevertTarget::Commit(state_hash),
)
.await
{
Expand Down Expand Up @@ -360,11 +363,28 @@ impl<DB: database::DB, VM: vm::VMExecution, N: Network> InSyncImpl<DB, VM, N> {
new_iter = remote_blk.header().iteration,
);

let state_hash = acc
.db
.read()
.await
.view(|t| {
let res = t
.fetch_block_header(
&remote_blk.header().prev_block_hash,
)?
.map(|(prev_header, _)| prev_header.state_hash);

anyhow::Ok::<Option<[u8; 32]>>(res)
})?
.ok_or_else(|| {
anyhow::anyhow!("could not retrieve state_hash")
})?;

match fallback::WithContext::new(acc.deref())
.try_revert(
&local_header,
remote_blk.header(),
RevertTarget::LastFinalizedState,
RevertTarget::Commit(state_hash),
)
.await
{
Expand Down
6 changes: 5 additions & 1 deletion node/src/vm.rs
Original file line number Diff line number Diff line change
Expand Up @@ -48,5 +48,9 @@ pub trait VMExecution: Send + Sync + 'static {

fn get_state_root(&self) -> anyhow::Result<[u8; 32]>;

fn revert(&self) -> anyhow::Result<[u8; 32]>;
/// Returns last finalized state root
fn get_finalized_state_root(&self) -> anyhow::Result<[u8; 32]>;

fn revert(&self, state_hash: [u8; 32]) -> anyhow::Result<[u8; 32]>;
fn revert_to_finalized(&self) -> anyhow::Result<[u8; 32]>;
}
2 changes: 1 addition & 1 deletion node/testbed.sh
Original file line number Diff line number Diff line change
Expand Up @@ -51,7 +51,7 @@
RUSK_STATE_PATH=${RUSK_STATE_PATH} cargo r --release -p rusk -- recovery-state --init $GENESIS_PATH
echo "starting node $ID ..."
echo "${KEYS_PATH}/node_$ID.keys"
RUSK_STATE_PATH=${RUSK_STATE_PATH} ./target/release/rusk --kadcast-bootstrap "$BOOTSTRAP_ADDR" --kadcast-public-address "$PUBLIC_ADDR" --log-level="$LOG_LEVEL" --log-filter="dusk_consensus=debug" --consensus-keys-path="${KEYS_PATH}/node_$ID.keys" --db-path="$NODE_FOLDER" --http-listen-addr "$WS_LISTEN_ADDR" --delay-on-resp-msg=10 > "${TEMPD}/node_${ID}.log" &
RUSK_STATE_PATH=${RUSK_STATE_PATH} ./target/release/rusk --kadcast-bootstrap "$BOOTSTRAP_ADDR" --kadcast-public-address "$PUBLIC_ADDR" --log-type="json" --log-level="$LOG_LEVEL" --log-filter="dusk_consensus=debug" --consensus-keys-path="${KEYS_PATH}/node_$ID.keys" --db-path="$NODE_FOLDER" --http-listen-addr "$WS_LISTEN_ADDR" --delay-on-resp-msg=10 > "${TEMPD}/node_${ID}.log" &
}

## Use ~/.cargo/bin/tokio-console --retain-for 0s http://127.0.0.1:10000 to connect console to first node
Expand Down
3 changes: 2 additions & 1 deletion rusk/benches/block_ingestion.rs
Original file line number Diff line number Diff line change
Expand Up @@ -99,7 +99,8 @@ pub fn accept_benchmark(c: &mut Criterion) {
)
.expect("Accepting transactions should succeed");

rusk.revert().expect("Reverting should succeed");
rusk.revert_to_base_root()
.expect("Reverting should succeed");
})
},
);
Expand Down
5 changes: 5 additions & 0 deletions rusk/src/lib/error.rs
Original file line number Diff line number Diff line change
Expand Up @@ -49,6 +49,8 @@ pub enum Error {
InconsistentState(VerificationOutput),
/// Other
Other(Box<dyn std::error::Error>),
/// Commit not found amongst existing commits
CommitNotFound([u8; 32]),
}

impl std::error::Error for Error {}
Expand Down Expand Up @@ -140,6 +142,9 @@ impl fmt::Display for Error {
"Inconsistent state verification data {verification_output}",
)
}
Error::CommitNotFound(commit_id) => {
write!(f, "Commit not found, id = {}", hex::encode(commit_id),)
}
}
}
}
14 changes: 12 additions & 2 deletions rusk/src/lib/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -334,12 +334,22 @@ impl Rusk {
Ok(())
}

pub fn revert(&self) -> Result<[u8; 32]> {
pub fn revert(&self, state_hash: [u8; 32]) -> Result<[u8; 32]> {
let mut inner = self.inner.lock();
inner.current_commit = inner.base_commit;

let commits = &inner.vm.commits();
if !commits.contains(&state_hash) {
return Err(Error::CommitNotFound(state_hash));
}

inner.current_commit = state_hash;
Ok(inner.current_commit)
}

pub fn revert_to_base_root(&self) -> Result<[u8; 32]> {
self.revert(self.base_root())
}

/// Perform an action with the underlying data structure.
pub fn with_inner<'a, F, T>(&'a self, closure: F) -> T
where
Expand Down
16 changes: 14 additions & 2 deletions rusk/src/lib/vm.rs
Original file line number Diff line number Diff line change
Expand Up @@ -149,13 +149,25 @@ impl VMExecution for Rusk {
Ok(self.state_root())
}

fn revert(&self) -> anyhow::Result<[u8; 32]> {
fn get_finalized_state_root(&self) -> anyhow::Result<[u8; 32]> {
Ok(self.base_root())
}

fn revert(&self, state_hash: [u8; 32]) -> anyhow::Result<[u8; 32]> {
let state_hash = self
.revert()
.revert(state_hash)
.map_err(|inner| anyhow::anyhow!("Cannot revert: {inner}"))?;

Ok(state_hash)
}

fn revert_to_finalized(&self) -> anyhow::Result<[u8; 32]> {
let state_hash = self.revert_to_base_root().map_err(|inner| {
anyhow::anyhow!("Cannot revert to finalized: {inner}")
})?;

Ok(state_hash)
}
}

impl Rusk {
Expand Down
4 changes: 2 additions & 2 deletions rusk/tests/rusk-state.rs
Original file line number Diff line number Diff line change
Expand Up @@ -102,7 +102,7 @@ pub fn rusk_state_accepted() -> Result<()> {
"There should be two notes in the state now"
);

rusk.revert()?;
rusk.revert_to_base_root()?;
let leaves = leaves_from_height(&rusk, 0)?;

assert_eq!(
Expand Down Expand Up @@ -134,7 +134,7 @@ pub fn rusk_state_finalized() -> Result<()> {
"There should be two notes in the state now"
);

rusk.revert()?;
rusk.revert_to_base_root()?;
let leaves = leaves_from_height(&rusk, 0)?;

assert_eq!(
Expand Down
3 changes: 2 additions & 1 deletion rusk/tests/services/transfer.rs
Original file line number Diff line number Diff line change
Expand Up @@ -185,7 +185,8 @@ pub async fn wallet() -> Result<()> {
assert_ne!(original_root, new_root, "Root should have changed");

// Revert the state
rusk.revert().expect("Reverting should succeed");
rusk.revert_to_base_root()
.expect("Reverting should succeed");
cache.write().unwrap().clear();

// Check the state's root is back to the original one
Expand Down