From a9ea21a44999695e5b033cdeb1a2106cc85b5c8c Mon Sep 17 00:00:00 2001 From: Shanin Roman <40040452+Erigara@users.noreply.github.com> Date: Tue, 1 Oct 2024 03:11:24 +0300 Subject: [PATCH] fix(block_sync): check that next block has height +1 (#5111) Signed-off-by: Shanin Roman --- crates/iroha_core/src/block_sync.rs | 123 +++++++++++++++++++++++++--- 1 file changed, 113 insertions(+), 10 deletions(-) diff --git a/crates/iroha_core/src/block_sync.rs b/crates/iroha_core/src/block_sync.rs index da415e3096b..c7ea19eb4d0 100644 --- a/crates/iroha_core/src/block_sync.rs +++ b/crates/iroha_core/src/block_sync.rs @@ -328,6 +328,12 @@ pub mod message { blocks: Vec, } + enum ShareBlocksError { + HeightMissed, + PrevBlockHashMismatch, + Empty, + } + impl GetBlocksAfterCandidate { fn validate(self) -> Result { if self.prev_hash.is_some() && self.latest_hash.is_none() { @@ -345,20 +351,34 @@ pub mod message { } } + impl From for parity_scale_codec::Error { + fn from(value: ShareBlocksError) -> Self { + match value { + ShareBlocksError::Empty => "Blocks are empty", + ShareBlocksError::HeightMissed => "There is a gap between blocks", + ShareBlocksError::PrevBlockHashMismatch => { + "Mismatch between previous block in the header and actual hash" + } + } + .into() + } + } + impl ShareBlocksCandidate { - fn validate(self) -> Result { + fn validate(self) -> Result { if self.blocks.is_empty() { - return Err(parity_scale_codec::Error::from("Blocks are empty")); + return Err(ShareBlocksError::Empty); } - if !self.blocks.windows(2).all(|wnd| { - wnd[1].header().height.get() == wnd[0].header().height.get() - 1 - && wnd[1].header().prev_block_hash == Some(wnd[0].hash()) - }) { - return Err(parity_scale_codec::Error::from( - "Blocks are not ordered correctly", - )); - } + self.blocks.windows(2).try_for_each(|wnd| { + if wnd[1].header().height.get() != wnd[0].header().height.get() + 1 { + return Err(ShareBlocksError::HeightMissed); + } + if wnd[1].header().prev_block_hash != Some(wnd[0].hash()) { + return Err(ShareBlocksError::PrevBlockHashMismatch); + } + Ok(()) + })?; Ok(ShareBlocks { peer_id: self.peer, @@ -382,5 +402,88 @@ pub mod message { .map_err(Into::into) } } + + #[cfg(test)] + mod tests { + use iroha_crypto::{Hash, KeyPair}; + + use super::*; + use crate::block::ValidBlock; + + #[test] + fn candidate_empty() { + let (leader_public_key, _) = KeyPair::random().into_parts(); + let leader_peer_id = + PeerId::new("127.0.0.1:1234".parse().unwrap(), leader_public_key); + let candidate = ShareBlocksCandidate { + blocks: Vec::new(), + peer: leader_peer_id, + }; + assert!(matches!(candidate.validate(), Err(ShareBlocksError::Empty))) + } + + #[test] + fn candidate_height_missed() { + let (leader_public_key, leader_private_key) = KeyPair::random().into_parts(); + let leader_peer_id = + PeerId::new("127.0.0.1:1234".parse().unwrap(), leader_public_key); + let block0: SignedBlock = ValidBlock::new_dummy(&leader_private_key).into(); + let block1 = + ValidBlock::new_dummy_and_modify_payload(&leader_private_key, |payload| { + payload.header.height = block0.header().height.checked_add(2).unwrap(); + }) + .into(); + let candidate = ShareBlocksCandidate { + blocks: vec![block0, block1], + peer: leader_peer_id, + }; + assert!(matches!( + candidate.validate(), + Err(ShareBlocksError::HeightMissed) + )) + } + + #[test] + fn candidate_prev_block_hash_mismatch() { + let (leader_public_key, leader_private_key) = KeyPair::random().into_parts(); + let leader_peer_id = + PeerId::new("127.0.0.1:1234".parse().unwrap(), leader_public_key); + let block0: SignedBlock = ValidBlock::new_dummy(&leader_private_key).into(); + let block1 = + ValidBlock::new_dummy_and_modify_payload(&leader_private_key, |payload| { + payload.header.height = block0.header().height.checked_add(1).unwrap(); + payload.header.prev_block_hash = + Some(HashOf::from_untyped_unchecked(Hash::prehashed([0; 32]))); + }) + .into(); + let candidate = ShareBlocksCandidate { + blocks: vec![block0, block1], + peer: leader_peer_id, + }; + assert!(matches!( + candidate.validate(), + Err(ShareBlocksError::PrevBlockHashMismatch) + )) + } + + #[test] + fn candidate_ok() { + let (leader_public_key, leader_private_key) = KeyPair::random().into_parts(); + let leader_peer_id = + PeerId::new("127.0.0.1:1234".parse().unwrap(), leader_public_key); + let block0: SignedBlock = ValidBlock::new_dummy(&leader_private_key).into(); + let block1 = + ValidBlock::new_dummy_and_modify_payload(&leader_private_key, |payload| { + payload.header.height = block0.header().height.checked_add(1).unwrap(); + payload.header.prev_block_hash = Some(block0.hash()); + }) + .into(); + let candidate = ShareBlocksCandidate { + blocks: vec![block0, block1], + peer: leader_peer_id, + }; + assert!(candidate.validate().is_ok()) + } + } } }