// Copyright 2015-2019 Parity Technologies (UK) Ltd. // This file is part of Parity Ethereum. // Parity Ethereum is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // Parity Ethereum is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // You should have received a copy of the GNU General Public License // along with Parity Ethereum. If not, see . //! Import route. use ethereum_types::H256; use crate::block_info::{BlockInfo, BlockLocation}; /// Import route for newly inserted block. #[derive(Debug, PartialEq, Clone)] pub struct ImportRoute { /// Blocks that were invalidated by new block. pub retracted: Vec, /// Blocks that were validated by new block. pub enacted: Vec, /// Blocks which are neither retracted nor enacted. pub omitted: Vec, } impl ImportRoute { /// Empty import route. pub fn none() -> Self { ImportRoute { retracted: vec![], enacted: vec![], omitted: vec![], } } } impl From for ImportRoute { fn from(info: BlockInfo) -> ImportRoute { match info.location { BlockLocation::CanonChain => ImportRoute { retracted: vec![], enacted: vec![info.hash], omitted: vec![], }, BlockLocation::Branch => ImportRoute { retracted: vec![], enacted: vec![], omitted: vec![info.hash], }, BlockLocation::BranchBecomingCanonChain(mut data) => { data.enacted.push(info.hash); ImportRoute { retracted: data.retracted, enacted: data.enacted, omitted: vec![], } } } } } #[cfg(test)] mod tests { use ethereum_types::{U256, BigEndianHash}; use crate::block_info::{BlockInfo, BlockLocation, BranchBecomingCanonChainData}; use super::ImportRoute; #[test] fn import_route_none() { assert_eq!(ImportRoute::none(), ImportRoute { enacted: vec![], retracted: vec![], omitted: vec![], }); } #[test] fn import_route_branch() { let info = BlockInfo { hash: BigEndianHash::from_uint(&U256::from(1)), number: 0, total_difficulty: U256::from(0), location: BlockLocation::Branch, }; assert_eq!(ImportRoute::from(info), ImportRoute { retracted: vec![], enacted: vec![], omitted: vec![BigEndianHash::from_uint(&U256::from(1))], }); } #[test] fn import_route_canon_chain() { let info = BlockInfo { hash: BigEndianHash::from_uint(&U256::from(1)), number: 0, total_difficulty: U256::from(0), location: BlockLocation::CanonChain, }; assert_eq!(ImportRoute::from(info), ImportRoute { retracted: vec![], enacted: vec![BigEndianHash::from_uint(&U256::from(1))], omitted: vec![], }); } #[test] fn import_route_branch_becoming_canon_chain() { let info = BlockInfo { hash: BigEndianHash::from_uint(&U256::from(2)), number: 0, total_difficulty: U256::from(0), location: BlockLocation::BranchBecomingCanonChain(BranchBecomingCanonChainData { ancestor: BigEndianHash::from_uint(&U256::from(0)), enacted: vec![BigEndianHash::from_uint(&U256::from(1))], retracted: vec![BigEndianHash::from_uint(&U256::from(3)), BigEndianHash::from_uint(&U256::from(4))], }) }; assert_eq!(ImportRoute::from(info), ImportRoute { retracted: vec![BigEndianHash::from_uint(&U256::from(3)), BigEndianHash::from_uint(&U256::from(4))], enacted: vec![BigEndianHash::from_uint(&U256::from(1)), BigEndianHash::from_uint(&U256::from(2))], omitted: vec![], }); } }