mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-05-30 03:31:05 +00:00
0d6884b919
* primitives/runtime: initial changes on supporting multiple Justifications * primitives/runtime: make Justifications strongly typed * Encode/decode Justifications * primitives/runtime: add Justification type * backend: apply_finality and finalize_block takes a single Justification * manual-seal: create engine id and let rpc take encoded justification * backend: skeleton functions for appending justifications * backend: initial implementation append_justification Initial implementation of append_justification on the Backend trait, and also remove unused skeleton functions for append_justificaton on Finaziler trait. k * backend: guard against duplicate consensus engine id * client/db: add check for block finality * client/api: add append_justification to in_mem db * client/light: add no-op append_justification * network: fix decode call for Justification * network: only send a single Justification in BlockData * network: minor comment update * protocol: update field names to distinguish single justification * client: further field renames to plural * client: update function names to plural justifications * client/db: upgrade existing database for new format * network: remove dependency on grandpa crate * db: fix check for finalized block * grandpa: check for multiple grandpa justifications hwne importing * backend: update Finalizer trait to take multiple Justifications * db: remove debugging statements in migration code * manual-seal: update note about engine id * db: fix check for finalized block * client: update variable name to reflect it is now plural * grandpa: fix incorrect empty Justications in test * primitives: make Justifications opaque to avoid being empty * network: fix detecting empty Justification * runtime: doc strings for Justifications functions * runtime: add into_justifications * primitives: check for duplicates in when adding to Justifications * network/test: use real grandpa engine id in test * client: fix reviewer comments * primitives: rename Justifications::push to append * backend: revert changes to Finalizer trait * backend: revert mark_finalized * backend: revert changes to finalize_block * backend: revert finalized_blocks * db: add a quick early return for performance * client: minor reviewer comments * service/test: use local ConsensusEngineId * network: add link to issue for sending multiple Justifications * Apply suggestions from code review Co-authored-by: Pierre Krieger <pierre.krieger1708@gmail.com> * Apply suggestions from code review Co-authored-by: Pierre Krieger <pierre.krieger1708@gmail.com> * network: tweaks to review suggestions * network: revert change to BlockData for backwards compatibility * Apply suggestion from code review Co-authored-by: Pierre Krieger <pierre.krieger1708@gmail.com> * Apply suggestions from code review Co-authored-by: André Silva <123550+andresilva@users.noreply.github.com> * primitives: update doc comment for Justifications * client/db/upgrade: avoid grandpa crate dependency * consensus: revert to single Justification for import_justification * primitives: improve justifications docs * style cleanups * use and_then * client: rename JUSTIFICATIONS db column * network: revert to using FRNK in network-test Co-authored-by: Pierre Krieger <pierre.krieger1708@gmail.com> Co-authored-by: André Silva <123550+andresilva@users.noreply.github.com> Co-authored-by: André Silva <andrerfosilva@gmail.com>
543 lines
16 KiB
Rust
543 lines
16 KiB
Rust
// 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/>.
|
|
|
|
//! A manual sealing engine: the engine listens for rpc calls to seal blocks and create forks.
|
|
//! This is suitable for a testing environment.
|
|
|
|
use futures::prelude::*;
|
|
use sp_consensus::{
|
|
Environment, Proposer, SelectChain, BlockImport,
|
|
ForkChoiceStrategy, BlockImportParams, BlockOrigin,
|
|
import_queue::{Verifier, BasicQueue, CacheKeyId, BoxBlockImport},
|
|
};
|
|
use sp_blockchain::HeaderBackend;
|
|
use sp_inherents::InherentDataProviders;
|
|
use sp_runtime::{traits::Block as BlockT, Justifications, ConsensusEngineId};
|
|
use sc_client_api::backend::{Backend as ClientBackend, Finalizer};
|
|
use sc_transaction_pool::txpool;
|
|
use std::{sync::Arc, marker::PhantomData};
|
|
use prometheus_endpoint::Registry;
|
|
|
|
mod error;
|
|
mod finalize_block;
|
|
mod seal_block;
|
|
|
|
pub mod consensus;
|
|
pub mod rpc;
|
|
|
|
pub use self::{
|
|
error::Error,
|
|
consensus::ConsensusDataProvider,
|
|
finalize_block::{finalize_block, FinalizeBlockParams},
|
|
seal_block::{SealBlockParams, seal_block, MAX_PROPOSAL_DURATION},
|
|
rpc::{EngineCommand, CreatedBlock},
|
|
};
|
|
use sp_api::{ProvideRuntimeApi, TransactionFor};
|
|
|
|
/// The `ConsensusEngineId` of Manual Seal.
|
|
pub const MANUAL_SEAL_ENGINE_ID: ConsensusEngineId = [b'm', b'a', b'n', b'l'];
|
|
|
|
/// The verifier for the manual seal engine; instantly finalizes.
|
|
struct ManualSealVerifier;
|
|
|
|
impl<B: BlockT> Verifier<B> for ManualSealVerifier {
|
|
fn verify(
|
|
&mut self,
|
|
origin: BlockOrigin,
|
|
header: B::Header,
|
|
justifications: Option<Justifications>,
|
|
body: Option<Vec<B::Extrinsic>>,
|
|
) -> Result<(BlockImportParams<B, ()>, Option<Vec<(CacheKeyId, Vec<u8>)>>), String> {
|
|
let mut import_params = BlockImportParams::new(origin, header);
|
|
import_params.justifications = justifications;
|
|
import_params.body = body;
|
|
import_params.finalized = false;
|
|
import_params.fork_choice = Some(ForkChoiceStrategy::LongestChain);
|
|
|
|
Ok((import_params, None))
|
|
}
|
|
}
|
|
|
|
/// Instantiate the import queue for the manual seal consensus engine.
|
|
pub fn import_queue<Block, Transaction>(
|
|
block_import: BoxBlockImport<Block, Transaction>,
|
|
spawner: &impl sp_core::traits::SpawnEssentialNamed,
|
|
registry: Option<&Registry>,
|
|
) -> BasicQueue<Block, Transaction>
|
|
where
|
|
Block: BlockT,
|
|
Transaction: Send + Sync + 'static,
|
|
{
|
|
BasicQueue::new(
|
|
ManualSealVerifier,
|
|
block_import,
|
|
None,
|
|
spawner,
|
|
registry,
|
|
)
|
|
}
|
|
|
|
/// Params required to start the instant sealing authorship task.
|
|
pub struct ManualSealParams<B: BlockT, BI, E, C: ProvideRuntimeApi<B>, A: txpool::ChainApi, SC, CS> {
|
|
/// Block import instance for well. importing blocks.
|
|
pub block_import: BI,
|
|
|
|
/// The environment we are producing blocks for.
|
|
pub env: E,
|
|
|
|
/// Client instance
|
|
pub client: Arc<C>,
|
|
|
|
/// Shared reference to the transaction pool.
|
|
pub pool: Arc<txpool::Pool<A>>,
|
|
|
|
/// Stream<Item = EngineCommands>, Basically the receiving end of a channel for sending commands to
|
|
/// the authorship task.
|
|
pub commands_stream: CS,
|
|
|
|
/// SelectChain strategy.
|
|
pub select_chain: SC,
|
|
|
|
/// Digest provider for inclusion in blocks.
|
|
pub consensus_data_provider: Option<Box<dyn ConsensusDataProvider<B, Transaction = TransactionFor<C, B>>>>,
|
|
|
|
/// Provider for inherents to include in blocks.
|
|
pub inherent_data_providers: InherentDataProviders,
|
|
}
|
|
|
|
/// Params required to start the manual sealing authorship task.
|
|
pub struct InstantSealParams<B: BlockT, BI, E, C: ProvideRuntimeApi<B>, A: txpool::ChainApi, SC> {
|
|
/// Block import instance for well. importing blocks.
|
|
pub block_import: BI,
|
|
|
|
/// The environment we are producing blocks for.
|
|
pub env: E,
|
|
|
|
/// Client instance
|
|
pub client: Arc<C>,
|
|
|
|
/// Shared reference to the transaction pool.
|
|
pub pool: Arc<txpool::Pool<A>>,
|
|
|
|
/// SelectChain strategy.
|
|
pub select_chain: SC,
|
|
|
|
/// Digest provider for inclusion in blocks.
|
|
pub consensus_data_provider: Option<Box<dyn ConsensusDataProvider<B, Transaction = TransactionFor<C, B>>>>,
|
|
|
|
/// Provider for inherents to include in blocks.
|
|
pub inherent_data_providers: InherentDataProviders,
|
|
}
|
|
|
|
/// Creates the background authorship task for the manual seal engine.
|
|
pub async fn run_manual_seal<B, BI, CB, E, C, A, SC, CS>(
|
|
ManualSealParams {
|
|
mut block_import,
|
|
mut env,
|
|
client,
|
|
pool,
|
|
mut commands_stream,
|
|
select_chain,
|
|
inherent_data_providers,
|
|
consensus_data_provider,
|
|
..
|
|
}: ManualSealParams<B, BI, E, C, A, SC, CS>
|
|
)
|
|
where
|
|
A: txpool::ChainApi<Block=B> + 'static,
|
|
B: BlockT + 'static,
|
|
BI: BlockImport<B, Error = sp_consensus::Error, Transaction = sp_api::TransactionFor<C, B>>
|
|
+ Send + Sync + 'static,
|
|
C: HeaderBackend<B> + Finalizer<B, CB> + ProvideRuntimeApi<B> + 'static,
|
|
CB: ClientBackend<B> + 'static,
|
|
E: Environment<B> + 'static,
|
|
E::Proposer: Proposer<B, Transaction = TransactionFor<C, B>>,
|
|
CS: Stream<Item=EngineCommand<<B as BlockT>::Hash>> + Unpin + 'static,
|
|
SC: SelectChain<B> + 'static,
|
|
TransactionFor<C, B>: 'static,
|
|
{
|
|
while let Some(command) = commands_stream.next().await {
|
|
match command {
|
|
EngineCommand::SealNewBlock {
|
|
create_empty,
|
|
finalize,
|
|
parent_hash,
|
|
sender,
|
|
} => {
|
|
seal_block(
|
|
SealBlockParams {
|
|
sender,
|
|
parent_hash,
|
|
finalize,
|
|
create_empty,
|
|
env: &mut env,
|
|
select_chain: &select_chain,
|
|
block_import: &mut block_import,
|
|
inherent_data_provider: &inherent_data_providers,
|
|
consensus_data_provider: consensus_data_provider.as_ref().map(|p| &**p),
|
|
pool: pool.clone(),
|
|
client: client.clone(),
|
|
}
|
|
).await;
|
|
}
|
|
EngineCommand::FinalizeBlock { hash, sender, justification } => {
|
|
let justification = justification.map(|j| (MANUAL_SEAL_ENGINE_ID, j));
|
|
finalize_block(
|
|
FinalizeBlockParams {
|
|
hash,
|
|
sender,
|
|
justification,
|
|
finalizer: client.clone(),
|
|
_phantom: PhantomData,
|
|
}
|
|
).await
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/// runs the background authorship task for the instant seal engine.
|
|
/// instant-seal creates a new block for every transaction imported into
|
|
/// the transaction pool.
|
|
pub async fn run_instant_seal<B, BI, CB, E, C, A, SC>(
|
|
InstantSealParams {
|
|
block_import,
|
|
env,
|
|
client,
|
|
pool,
|
|
select_chain,
|
|
consensus_data_provider,
|
|
inherent_data_providers,
|
|
..
|
|
}: InstantSealParams<B, BI, E, C, A, SC>
|
|
)
|
|
where
|
|
A: txpool::ChainApi<Block=B> + 'static,
|
|
B: BlockT + 'static,
|
|
BI: BlockImport<B, Error = sp_consensus::Error, Transaction = sp_api::TransactionFor<C, B>>
|
|
+ Send + Sync + 'static,
|
|
C: HeaderBackend<B> + Finalizer<B, CB> + ProvideRuntimeApi<B> + 'static,
|
|
CB: ClientBackend<B> + 'static,
|
|
E: Environment<B> + 'static,
|
|
E::Proposer: Proposer<B, Transaction = TransactionFor<C, B>>,
|
|
SC: SelectChain<B> + 'static,
|
|
TransactionFor<C, B>: 'static,
|
|
{
|
|
// instant-seal creates blocks as soon as transactions are imported
|
|
// into the transaction pool.
|
|
let commands_stream = pool.validated_pool()
|
|
.import_notification_stream()
|
|
.map(|_| {
|
|
EngineCommand::SealNewBlock {
|
|
create_empty: false,
|
|
finalize: false,
|
|
parent_hash: None,
|
|
sender: None,
|
|
}
|
|
});
|
|
|
|
run_manual_seal(
|
|
ManualSealParams {
|
|
block_import,
|
|
env,
|
|
client,
|
|
pool,
|
|
commands_stream,
|
|
select_chain,
|
|
consensus_data_provider,
|
|
inherent_data_providers,
|
|
}
|
|
).await
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use super::*;
|
|
use substrate_test_runtime_client::{
|
|
DefaultTestClientBuilderExt,
|
|
TestClientBuilderExt,
|
|
AccountKeyring::*,
|
|
TestClientBuilder,
|
|
};
|
|
use sc_transaction_pool::{BasicPool, RevalidationType, txpool::Options};
|
|
use substrate_test_runtime_transaction_pool::{TestApi, uxt};
|
|
use sp_transaction_pool::{TransactionPool, MaintainedTransactionPool, TransactionSource};
|
|
use sp_runtime::generic::BlockId;
|
|
use sp_consensus::ImportedAux;
|
|
use sp_inherents::InherentDataProviders;
|
|
use sc_basic_authorship::ProposerFactory;
|
|
use sc_client_api::BlockBackend;
|
|
|
|
fn api() -> Arc<TestApi> {
|
|
Arc::new(TestApi::empty())
|
|
}
|
|
|
|
const SOURCE: TransactionSource = TransactionSource::External;
|
|
|
|
#[tokio::test]
|
|
async fn instant_seal() {
|
|
let builder = TestClientBuilder::new();
|
|
let (client, select_chain) = builder.build_with_longest_chain();
|
|
let client = Arc::new(client);
|
|
let inherent_data_providers = InherentDataProviders::new();
|
|
let spawner = sp_core::testing::TaskExecutor::new();
|
|
let pool = Arc::new(BasicPool::with_revalidation_type(
|
|
Options::default(), true.into(), api(), None, RevalidationType::Full, spawner.clone(),
|
|
));
|
|
let env = ProposerFactory::new(
|
|
spawner.clone(),
|
|
client.clone(),
|
|
pool.clone(),
|
|
None,
|
|
None,
|
|
);
|
|
// this test checks that blocks are created as soon as transactions are imported into the pool.
|
|
let (sender, receiver) = futures::channel::oneshot::channel();
|
|
let mut sender = Arc::new(Some(sender));
|
|
let commands_stream = pool.pool().validated_pool().import_notification_stream()
|
|
.map(move |_| {
|
|
// we're only going to submit one tx so this fn will only be called once.
|
|
let mut_sender = Arc::get_mut(&mut sender).unwrap();
|
|
let sender = std::mem::take(mut_sender);
|
|
EngineCommand::SealNewBlock {
|
|
create_empty: false,
|
|
finalize: true,
|
|
parent_hash: None,
|
|
sender
|
|
}
|
|
});
|
|
let future = run_manual_seal(
|
|
ManualSealParams {
|
|
block_import: client.clone(),
|
|
env,
|
|
client: client.clone(),
|
|
pool: pool.pool().clone(),
|
|
commands_stream,
|
|
select_chain,
|
|
inherent_data_providers,
|
|
consensus_data_provider: None,
|
|
}
|
|
);
|
|
std::thread::spawn(|| {
|
|
let mut rt = tokio::runtime::Runtime::new().unwrap();
|
|
// spawn the background authorship task
|
|
rt.block_on(future);
|
|
});
|
|
// submit a transaction to pool.
|
|
let result = pool.submit_one(&BlockId::Number(0), SOURCE, uxt(Alice, 0)).await;
|
|
// assert that it was successfully imported
|
|
assert!(result.is_ok());
|
|
// assert that the background task returns ok
|
|
let created_block = receiver.await.unwrap().unwrap();
|
|
assert_eq!(
|
|
created_block,
|
|
CreatedBlock {
|
|
hash: created_block.hash.clone(),
|
|
aux: ImportedAux {
|
|
header_only: false,
|
|
clear_justification_requests: false,
|
|
needs_justification: false,
|
|
bad_justification: false,
|
|
is_new_best: true,
|
|
}
|
|
}
|
|
);
|
|
// assert that there's a new block in the db.
|
|
assert!(client.header(&BlockId::Number(1)).unwrap().is_some())
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn manual_seal_and_finalization() {
|
|
let builder = TestClientBuilder::new();
|
|
let (client, select_chain) = builder.build_with_longest_chain();
|
|
let client = Arc::new(client);
|
|
let inherent_data_providers = InherentDataProviders::new();
|
|
let spawner = sp_core::testing::TaskExecutor::new();
|
|
let pool = Arc::new(BasicPool::with_revalidation_type(
|
|
Options::default(), true.into(), api(), None, RevalidationType::Full, spawner.clone(),
|
|
));
|
|
let env = ProposerFactory::new(
|
|
spawner.clone(),
|
|
client.clone(),
|
|
pool.clone(),
|
|
None,
|
|
None,
|
|
);
|
|
// this test checks that blocks are created as soon as an engine command is sent over the stream.
|
|
let (mut sink, commands_stream) = futures::channel::mpsc::channel(1024);
|
|
let future = run_manual_seal(
|
|
ManualSealParams {
|
|
block_import: client.clone(),
|
|
env,
|
|
client: client.clone(),
|
|
pool: pool.pool().clone(),
|
|
commands_stream,
|
|
select_chain,
|
|
consensus_data_provider: None,
|
|
inherent_data_providers,
|
|
}
|
|
);
|
|
std::thread::spawn(|| {
|
|
let mut rt = tokio::runtime::Runtime::new().unwrap();
|
|
// spawn the background authorship task
|
|
rt.block_on(future);
|
|
});
|
|
// submit a transaction to pool.
|
|
let result = pool.submit_one(&BlockId::Number(0), SOURCE, uxt(Alice, 0)).await;
|
|
// assert that it was successfully imported
|
|
assert!(result.is_ok());
|
|
let (tx, rx) = futures::channel::oneshot::channel();
|
|
sink.send(EngineCommand::SealNewBlock {
|
|
parent_hash: None,
|
|
sender: Some(tx),
|
|
create_empty: false,
|
|
finalize: false,
|
|
}).await.unwrap();
|
|
let created_block = rx.await.unwrap().unwrap();
|
|
|
|
// assert that the background task returns ok
|
|
assert_eq!(
|
|
created_block,
|
|
CreatedBlock {
|
|
hash: created_block.hash.clone(),
|
|
aux: ImportedAux {
|
|
header_only: false,
|
|
clear_justification_requests: false,
|
|
needs_justification: false,
|
|
bad_justification: false,
|
|
is_new_best: true,
|
|
}
|
|
}
|
|
);
|
|
// assert that there's a new block in the db.
|
|
let header = client.header(&BlockId::Number(1)).unwrap().unwrap();
|
|
let (tx, rx) = futures::channel::oneshot::channel();
|
|
sink.send(EngineCommand::FinalizeBlock {
|
|
sender: Some(tx),
|
|
hash: header.hash(),
|
|
justification: None
|
|
}).await.unwrap();
|
|
// assert that the background task returns ok
|
|
assert_eq!(rx.await.unwrap().unwrap(), ());
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn manual_seal_fork_blocks() {
|
|
let builder = TestClientBuilder::new();
|
|
let (client, select_chain) = builder.build_with_longest_chain();
|
|
let client = Arc::new(client);
|
|
let inherent_data_providers = InherentDataProviders::new();
|
|
let pool_api = api();
|
|
let spawner = sp_core::testing::TaskExecutor::new();
|
|
let pool = Arc::new(BasicPool::with_revalidation_type(
|
|
Options::default(), true.into(), pool_api.clone(), None, RevalidationType::Full, spawner.clone(),
|
|
));
|
|
let env = ProposerFactory::new(
|
|
spawner.clone(),
|
|
client.clone(),
|
|
pool.clone(),
|
|
None,
|
|
None,
|
|
);
|
|
// this test checks that blocks are created as soon as an engine command is sent over the stream.
|
|
let (mut sink, commands_stream) = futures::channel::mpsc::channel(1024);
|
|
let future = run_manual_seal(
|
|
ManualSealParams {
|
|
block_import: client.clone(),
|
|
env,
|
|
client: client.clone(),
|
|
pool: pool.pool().clone(),
|
|
commands_stream,
|
|
select_chain,
|
|
consensus_data_provider: None,
|
|
inherent_data_providers,
|
|
}
|
|
);
|
|
std::thread::spawn(|| {
|
|
let mut rt = tokio::runtime::Runtime::new().unwrap();
|
|
// spawn the background authorship task
|
|
rt.block_on(future);
|
|
});
|
|
// submit a transaction to pool.
|
|
let result = pool.submit_one(&BlockId::Number(0), SOURCE, uxt(Alice, 0)).await;
|
|
// assert that it was successfully imported
|
|
assert!(result.is_ok());
|
|
|
|
let (tx, rx) = futures::channel::oneshot::channel();
|
|
sink.send(EngineCommand::SealNewBlock {
|
|
parent_hash: None,
|
|
sender: Some(tx),
|
|
create_empty: false,
|
|
finalize: false,
|
|
}).await.unwrap();
|
|
let created_block = rx.await.unwrap().unwrap();
|
|
pool_api.increment_nonce(Alice.into());
|
|
|
|
// assert that the background task returns ok
|
|
assert_eq!(
|
|
created_block,
|
|
CreatedBlock {
|
|
hash: created_block.hash.clone(),
|
|
aux: ImportedAux {
|
|
header_only: false,
|
|
clear_justification_requests: false,
|
|
needs_justification: false,
|
|
bad_justification: false,
|
|
is_new_best: true
|
|
}
|
|
}
|
|
);
|
|
let block = client.block(&BlockId::Number(1)).unwrap().unwrap().block;
|
|
pool_api.add_block(block, true);
|
|
assert!(pool.submit_one(&BlockId::Number(1), SOURCE, uxt(Alice, 1)).await.is_ok());
|
|
|
|
let header = client.header(&BlockId::Number(1)).expect("db error").expect("imported above");
|
|
pool.maintain(sp_transaction_pool::ChainEvent::NewBestBlock {
|
|
hash: header.hash(),
|
|
tree_route: None,
|
|
}).await;
|
|
|
|
let (tx1, rx1) = futures::channel::oneshot::channel();
|
|
assert!(sink.send(EngineCommand::SealNewBlock {
|
|
parent_hash: Some(created_block.hash),
|
|
sender: Some(tx1),
|
|
create_empty: false,
|
|
finalize: false,
|
|
}).await.is_ok());
|
|
assert_matches::assert_matches!(
|
|
rx1.await.expect("should be no error receiving"),
|
|
Ok(_)
|
|
);
|
|
let block = client.block(&BlockId::Number(2)).unwrap().unwrap().block;
|
|
pool_api.add_block(block, true);
|
|
pool_api.increment_nonce(Alice.into());
|
|
|
|
assert!(pool.submit_one(&BlockId::Number(1), SOURCE, uxt(Alice, 2)).await.is_ok());
|
|
let (tx2, rx2) = futures::channel::oneshot::channel();
|
|
assert!(sink.send(EngineCommand::SealNewBlock {
|
|
parent_hash: Some(created_block.hash),
|
|
sender: Some(tx2),
|
|
create_empty: false,
|
|
finalize: false,
|
|
}).await.is_ok());
|
|
let imported = rx2.await.unwrap().unwrap();
|
|
// assert that fork block is in the db
|
|
assert!(client.header(&BlockId::Hash(imported.hash)).unwrap().is_some())
|
|
}
|
|
}
|