diff --git a/zebra-consensus/src/verify.rs b/zebra-consensus/src/verify.rs index 3b3158e94..c36668e80 100644 --- a/zebra-consensus/src/verify.rs +++ b/zebra-consensus/src/verify.rs @@ -13,6 +13,7 @@ use std::{ error, future::Future, pin::Pin, + sync::Arc, task::{Context, Poll}, }; use tower::{buffer::Buffer, Service}; @@ -22,32 +23,10 @@ use zebra_chain::block::{Block, BlockHeaderHash}; mod script; mod transaction; -/// The trait constraints that we expect from `zebra_state::ZebraState` errors. -type ZSE = Box; -/// The trait constraints that we expect from the `zebra_state::ZebraState` service. -/// `ZSF` is the `Future` type for `zebra_state::ZebraState`. This type is generic, -/// because `tower::Service` function calls require a `Sized` future type. -type ZS = Box< - dyn Service - + Send - + 'static, ->; - -/// Block verification service. -/// -/// After verification, blocks are added to `state_service`. We use a generic -/// future `ZSF` for that service, so that the underlying state service can be -/// wrapped in other services as needed. -struct BlockVerifier -where - ZSF: Future> + Send + 'static, -{ - state_service: ZS, +struct BlockVerifier { + state_service: S, } -/// The result type for the BlockVerifier Service. -type Response = BlockHeaderHash; - /// The error type for the BlockVerifier Service. // TODO(jlusby): Error = Report ? type Error = Box; @@ -55,11 +34,12 @@ type Error = Box; /// The BlockVerifier service implementation. /// /// After verification, blocks are added to the underlying state service. -impl Service for BlockVerifier +impl Service> for BlockVerifier where - ZSF: Future> + Send + 'static, + S: Service, + S::Future: Send + 'static, { - type Response = Response; + type Response = BlockHeaderHash; type Error = Error; type Future = Pin> + Send + 'static>>; @@ -68,8 +48,8 @@ where self.state_service.poll_ready(context) } - fn call(&mut self, block: Block) -> Self::Future { - let header_hash: BlockHeaderHash = (&block).into(); + fn call(&mut self, block: Arc) -> Self::Future { + let header_hash: BlockHeaderHash = block.as_ref().into(); // Ignore errors for now. // TODO(jlusby): Error = Report, handle errors from state_service. @@ -80,9 +60,9 @@ where // `state_service.call` is OK here because we already called // `state_service.poll_ready` in our `poll_ready`. - let add_block = self.state_service.call(zebra_state::Request::AddBlock { - block: block.into(), - }); + let add_block = self + .state_service + .call(zebra_state::Request::AddBlock { block }); async move { add_block.await?; @@ -92,25 +72,36 @@ where } } -/// Initialise the BlockVerifier service. +/// Return a block verification service, using the provided state service. /// -/// We use a generic type `ZS` for `state_service`, so that the -/// underlying state service can be wrapped in other services as needed. -/// For similar reasons, we also return a dynamic service type. -pub fn init( - state_service: ZS, +/// The block verifier holds a state service of type `S`, used as context for +/// block validation and to which newly verified blocks will be committed. This +/// state is pluggable to allow for testing or instrumentation. +/// +/// The returned type is opaque to allow instrumentation or other wrappers, but +/// can be boxed for storage. It is also `Clone` to allow sharing of a +/// verification service. +/// +/// This function should be called only once for a particular state service (and +/// the result be shared) rather than constructing multiple verification services +/// backed by the same state layer. +pub fn init( + state_service: S, ) -> impl Service< - Block, - Response = Response, + Arc, + Response = BlockHeaderHash, Error = Error, - Future = impl Future>, + Future = impl Future>, > + Send + Clone + 'static where - ZSF: Future> + Send + 'static, + S: Service + + Send + + 'static, + S::Future: Send + 'static, { - Buffer::new(BlockVerifier:: { state_service }, 1) + Buffer::new(BlockVerifier { state_service }, 1) } #[cfg(test)] @@ -138,20 +129,12 @@ mod tests { .init(); } - /// Initialise and return an unwrapped `BlockVerifier`. - fn init_block_verifier(state_service: ZS) -> BlockVerifier - where - ZSF: Future> + Send + 'static, - { - BlockVerifier:: { state_service } - } - #[tokio::test] #[spandoc::spandoc] async fn verify() -> Result<(), Report> { - let block = Block::zcash_deserialize(&zebra_test_vectors::BLOCK_MAINNET_415000_BYTES[..])?; - // TODO(teor): why does rustc say that _hash is unused? - let _hash: BlockHeaderHash = (&block).into(); + let block = + Arc::::zcash_deserialize(&zebra_test_vectors::BLOCK_MAINNET_415000_BYTES[..])?; + let hash: BlockHeaderHash = block.as_ref().into(); let state_service = Box::new(zebra_state::in_memory::init()); let mut block_verifier = super::init(state_service); @@ -165,7 +148,7 @@ mod tests { .map_err(|e| eyre!(e))?; ensure!( - matches!(verify_response, _hash), + verify_response == hash, "unexpected response kind: {:?}", verify_response ); @@ -178,12 +161,12 @@ mod tests { async fn round_trip() -> Result<(), Report> { install_tracing(); - let block = Block::zcash_deserialize(&zebra_test_vectors::BLOCK_MAINNET_GENESIS_BYTES[..])?; - // TODO(teor): why does rustc say that _hash is unused? - let _hash: BlockHeaderHash = (&block).into(); + let block = + Arc::::zcash_deserialize(&zebra_test_vectors::BLOCK_MAINNET_415000_BYTES[..])?; + let hash: BlockHeaderHash = block.as_ref().into(); - let state_service = Box::new(zebra_state::in_memory::init()); - let mut block_verifier = init_block_verifier(state_service); + let mut state_service = zebra_state::in_memory::init(); + let mut block_verifier = super::init(state_service.clone()); let verify_response = block_verifier .ready_and() @@ -194,24 +177,23 @@ mod tests { .map_err(|e| eyre!(e))?; ensure!( - matches!(verify_response, _hash), + verify_response == hash, "unexpected response kind: {:?}", verify_response ); - let state_response = block_verifier - .state_service + let state_response = state_service .ready_and() .await .map_err(|e| eyre!(e))? - .call(zebra_state::Request::GetBlock { hash: _hash }) + .call(zebra_state::Request::GetBlock { hash }) .await .map_err(|e| eyre!(e))?; match state_response { zebra_state::Response::Block { block: returned_block, - } => assert_eq!(&block, returned_block.as_ref()), + } => assert_eq!(block, returned_block), _ => bail!("unexpected response kind: {:?}", state_response), }