diff --git a/client/api/src/statekv.rs b/client/api/src/statekv.rs index 28ff59e..b698292 100644 --- a/client/api/src/statekv.rs +++ b/client/api/src/statekv.rs @@ -55,6 +55,8 @@ pub trait StateKv: Send + Sync { fn get_contract_tracing(&self, number: NumberFor, index: u32) -> Option; fn remove_contract_tracing(&self, number: NumberFor, index: u32) -> error::Result<()>; fn remove_contract_tracings_by_number(&self, number: NumberFor) -> error::Result<()>; + + fn revert_all(&self, number: NumberFor) -> error::Result<()>; } pub trait StateKvTransaction { @@ -173,4 +175,8 @@ impl> StateKv for Arc { fn remove_contract_tracings_by_number(&self, number: NumberFor) -> error::Result<()> { (&**self).remove_contract_tracings_by_number(number) } + + fn revert_all(&self, number: NumberFor) -> error::Result<()> { + (&**self).remove_contract_tracings_by_number(number) + } } diff --git a/client/db/src/lib.rs b/client/db/src/lib.rs index 45b6fdc..dd5ae99 100644 --- a/client/db/src/lib.rs +++ b/client/db/src/lib.rs @@ -430,6 +430,42 @@ impl ec_client_api::statekv::StateKv for StateKv { t.delete_prefix(columns::TRACING, prefix); }) } + + fn revert_all(&self, number: NumberFor) -> error::Result<()> { + let hash = >::get_hash(self, number).ok_or( + error::DatabaseError(format!("No hash for this number:{}", number).into()), + )?; + // state + >::delete_kvs_by_hash(self, hash)?; + + // child state + let prefix = hash.as_ref(); + let hash_len = prefix.len(); + let mut lookup_key = Vec::with_capacity(hash_len + 1); + lookup_key.extend(prefix); + lookup_key.push(SEPARATOR); + + let mut t = DBTransaction::with_capacity(1); + t.delete_prefix(columns::STATE_CHILD_KV, &lookup_key); + self.state_kv_db + .write(t) + .map_err(|e| error::DatabaseError(Box::new(e)))?; + + // extrinsic changes + let num_u64: u64 = number.saturated_into::(); + let prefix = &num_u64.to_le_bytes()[..]; + let mut t = DBTransaction::with_capacity(1); + t.delete_prefix(columns::EXTRINSIC_CHANGES, &prefix); + self.state_kv_db + .write(t) + .map_err(|e| error::DatabaseError(Box::new(e)))?; + + // contract tracing + >::remove_contract_tracings_by_number( + self, number, + )?; + Ok(()) + } } fn tracing_key(number: u64, index: u32) -> Vec { diff --git a/client/rpc/src/lib.rs b/client/rpc/src/lib.rs index ee9b693..ed965d6 100644 --- a/client/rpc/src/lib.rs +++ b/client/rpc/src/lib.rs @@ -12,13 +12,16 @@ use jsonrpc_derive::rpc; use sp_blockchain::HeaderBackend; use sp_core::Bytes; -use sp_runtime::traits::{Block as BlockT, BlockIdTo, Header}; +use sp_runtime::{ + generic::BlockId, + traits::{Block as BlockT, BlockIdTo, Header}, + SaturatedConversion, +}; use sp_utils::mpsc::{tracing_unbounded, TracingUnboundedReceiver, TracingUnboundedSender}; use ec_client_api::statekv; use error::EuropaRpcError; -use sp_runtime::generic::BlockId; pub enum Message { Forward(NumberOf), @@ -128,6 +131,15 @@ where self.backend .revert(diff, true) .map_err(error::client_err::)?; + let state_kv = self.client.state_kv(); + let mut current = best; + while current != height { + state_kv + .revert_all(current) + .map_err(|e| error::client_err::(e.into()))?; + current -= 1_u64.saturated_into(); + } + Ok(()) } fn state_kvs(