1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74
// This file is part of Substrate. // Copyright (C) 2020-2021 Parity Technologies (UK) Ltd. // SPDX-License-Identifier: GPL-3.0-or-later WITH Classpath-exception-2.0 // This program 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. // This program 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 this program. If not, see <https://www.gnu.org/licenses/>. //! Client fixed chain specification rules use std::collections::{HashMap, HashSet}; use sp_runtime::traits::{Block as BlockT, NumberFor}; use sc_client_api::{BadBlocks, ForkBlocks}; /// Chain specification rules lookup result. pub enum LookupResult<B: BlockT> { /// Specification rules do not contain any special rules about this block NotSpecial, /// The block is known to be bad and should not be imported KnownBad, /// There is a specified canonical block hash for the given height Expected(B::Hash), } /// Chain-specific block filtering rules. /// /// This holds known bad blocks and known good forks, and /// is usually part of the chain spec. pub struct BlockRules<B: BlockT> { bad: HashSet<B::Hash>, forks: HashMap<NumberFor<B>, B::Hash>, } impl<B: BlockT> BlockRules<B> { /// New block rules with provided black and white lists. pub fn new(fork_blocks: ForkBlocks<B>, bad_blocks: BadBlocks<B>) -> Self { Self { bad: bad_blocks.unwrap_or_else(|| HashSet::new()), forks: fork_blocks.unwrap_or_else(|| vec![]).into_iter().collect(), } } /// Mark a new block as bad. pub fn mark_bad(&mut self, hash: B::Hash) { self.bad.insert(hash); } /// Check if there's any rule affecting the given block. pub fn lookup(&self, number: NumberFor<B>, hash: &B::Hash) -> LookupResult<B> { if let Some(hash_for_height) = self.forks.get(&number) { if hash_for_height != hash { return LookupResult::Expected(hash_for_height.clone()) } } if self.bad.contains(hash) { return LookupResult::KnownBad } LookupResult::NotSpecial } }