mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-06-21 04:01:01 +00:00
258 lines
8.0 KiB
Rust
258 lines
8.0 KiB
Rust
// Copyright 2017 Parity Technologies (UK) Ltd.
|
|
// This file is part of Polkadot.
|
|
|
|
// Polkadot 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.
|
|
|
|
// Polkadot 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 Polkadot. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
//! Consensus service.
|
|
|
|
/// Consensus service. A long running service that manages BFT agreement and parachain
|
|
/// candidate agreement over the network.
|
|
///
|
|
/// This uses a handle to an underlying thread pool to dispatch heavy work
|
|
/// such as candidate verification while performing event-driven work
|
|
/// on a local event loop.
|
|
|
|
use std::thread;
|
|
use std::time::{Duration, Instant};
|
|
use std::sync::Arc;
|
|
|
|
use bft::{self, BftService};
|
|
use client::{BlockchainEvents, ChainHead, BlockBody};
|
|
use ed25519;
|
|
use futures::prelude::*;
|
|
use polkadot_api::LocalPolkadotApi;
|
|
use polkadot_primitives::{Block, Header};
|
|
use transaction_pool::TransactionPool;
|
|
use extrinsic_store::Store as ExtrinsicStore;
|
|
|
|
use tokio::executor::current_thread::TaskExecutor as LocalThreadHandle;
|
|
use tokio::runtime::TaskExecutor as ThreadPoolHandle;
|
|
use tokio::runtime::current_thread::Runtime as LocalRuntime;
|
|
use tokio::timer::{Delay, Interval};
|
|
|
|
use super::{Network, Collators, ProposerFactory};
|
|
use error;
|
|
|
|
const TIMER_DELAY_MS: u64 = 5000;
|
|
const TIMER_INTERVAL_MS: u64 = 500;
|
|
|
|
// spin up an instance of BFT agreement on the current thread's executor.
|
|
// panics if there is no current thread executor.
|
|
fn start_bft<F, C>(
|
|
header: Header,
|
|
bft_service: Arc<BftService<Block, F, C>>,
|
|
) where
|
|
F: bft::Environment<Block> + 'static,
|
|
C: bft::BlockImport<Block> + bft::Authorities<Block> + 'static,
|
|
F::Error: ::std::fmt::Debug,
|
|
<F::Proposer as bft::Proposer<Block>>::Error: ::std::fmt::Display + Into<error::Error>,
|
|
<F as bft::Environment<Block>>::Error: ::std::fmt::Display
|
|
{
|
|
const DELAY_UNTIL: Duration = Duration::from_millis(5000);
|
|
|
|
let mut handle = LocalThreadHandle::current();
|
|
let work = Delay::new(Instant::now() + DELAY_UNTIL)
|
|
.then(move |res| {
|
|
if let Err(e) = res {
|
|
warn!(target: "bft", "Failed to force delay of consensus: {:?}", e);
|
|
}
|
|
|
|
match bft_service.build_upon(&header) {
|
|
Ok(maybe_bft_work) => {
|
|
if maybe_bft_work.is_some() {
|
|
debug!(target: "bft", "Starting agreement. After forced delay for {:?}",
|
|
DELAY_UNTIL);
|
|
}
|
|
|
|
maybe_bft_work
|
|
}
|
|
Err(e) => {
|
|
warn!(target: "bft", "BFT agreement error: {}", e);
|
|
None
|
|
}
|
|
}
|
|
})
|
|
.map(|_| ());
|
|
|
|
if let Err(e) = handle.spawn_local(Box::new(work)) {
|
|
debug!(target: "bft", "Couldn't initialize BFT agreement: {:?}", e);
|
|
}
|
|
}
|
|
|
|
// creates a task to prune redundant entries in availability store upon block finalization
|
|
//
|
|
// NOTE: this will need to be changed to finality notification rather than
|
|
// block import notifications when the consensus switches to non-instant finality.
|
|
fn prune_unneeded_availability<C>(client: Arc<C>, extrinsic_store: ExtrinsicStore)
|
|
-> impl Future<Item=(),Error=()> + Send
|
|
where C: Send + Sync + BlockchainEvents<Block> + BlockBody<Block> + 'static
|
|
{
|
|
use codec::{Encode, Decode};
|
|
use polkadot_primitives::BlockId;
|
|
use polkadot_runtime::CheckedBlock;
|
|
|
|
enum NotifyError {
|
|
NoBody(::client::error::Error),
|
|
UnexpectedFormat,
|
|
ExtrinsicsWrong,
|
|
}
|
|
|
|
impl NotifyError {
|
|
fn log(&self, hash: &::polkadot_primitives::Hash) {
|
|
match *self {
|
|
NotifyError::NoBody(ref err) => warn!("Failed to fetch block body for imported block {:?}: {:?}", hash, err),
|
|
NotifyError::UnexpectedFormat => warn!("Consensus outdated: Block {:?} has unexpected body format", hash),
|
|
NotifyError::ExtrinsicsWrong => warn!("Consensus outdated: Failed to fetch block body for imported block {:?}", hash),
|
|
}
|
|
}
|
|
}
|
|
|
|
client.import_notification_stream()
|
|
.for_each(move |notification| {
|
|
let checked_block = client.block_body(&BlockId::hash(notification.hash))
|
|
.map_err(NotifyError::NoBody)
|
|
.map(|b| ::polkadot_runtime::Block::decode(&mut b.encode().as_slice()))
|
|
.and_then(|maybe_block| maybe_block.ok_or(NotifyError::UnexpectedFormat))
|
|
.and_then(|block| CheckedBlock::new(block).map_err(|_| NotifyError::ExtrinsicsWrong));
|
|
|
|
match checked_block {
|
|
Ok(block) => {
|
|
let candidate_hashes = block.parachain_heads().iter().map(|c| c.hash()).collect();
|
|
if let Err(e) = extrinsic_store.candidates_finalized(notification.header.parent_hash, candidate_hashes) {
|
|
warn!(target: "consensus", "Failed to prune unneeded available data: {:?}", e);
|
|
}
|
|
}
|
|
Err(e) => e.log(¬ification.hash)
|
|
}
|
|
|
|
Ok(())
|
|
})
|
|
}
|
|
|
|
/// Consensus service. Starts working when created.
|
|
pub struct Service {
|
|
thread: Option<thread::JoinHandle<()>>,
|
|
exit_signal: Option<::exit_future::Signal>,
|
|
}
|
|
|
|
impl Service {
|
|
/// Create and start a new instance.
|
|
pub fn new<A, C, N>(
|
|
client: Arc<C>,
|
|
api: Arc<A>,
|
|
network: N,
|
|
transaction_pool: Arc<TransactionPool<A>>,
|
|
thread_pool: ThreadPoolHandle,
|
|
parachain_empty_duration: Duration,
|
|
key: ed25519::Pair,
|
|
extrinsic_store: ExtrinsicStore,
|
|
) -> Service
|
|
where
|
|
A: LocalPolkadotApi + Send + Sync + 'static,
|
|
C: BlockchainEvents<Block> + ChainHead<Block> + BlockBody<Block> + bft::BlockImport<Block> + bft::Authorities<Block> + Send + Sync + 'static,
|
|
N: Network + Collators + Send + 'static,
|
|
N::TableRouter: Send + 'static,
|
|
<N::Collation as IntoFuture>::Future: Send + 'static,
|
|
{
|
|
let (signal, exit) = ::exit_future::signal();
|
|
let thread = thread::spawn(move || {
|
|
let mut runtime = LocalRuntime::new().expect("Could not create local runtime");
|
|
let key = Arc::new(key);
|
|
|
|
let factory = ProposerFactory {
|
|
client: api.clone(),
|
|
transaction_pool: transaction_pool.clone(),
|
|
collators: network.clone(),
|
|
network,
|
|
parachain_empty_duration,
|
|
handle: thread_pool.clone(),
|
|
extrinsic_store: extrinsic_store.clone(),
|
|
};
|
|
let bft_service = Arc::new(BftService::new(client.clone(), key, factory));
|
|
|
|
let notifications = {
|
|
let client = client.clone();
|
|
let bft_service = bft_service.clone();
|
|
|
|
client.import_notification_stream().for_each(move |notification| {
|
|
if notification.is_new_best {
|
|
start_bft(notification.header, bft_service.clone());
|
|
}
|
|
Ok(())
|
|
})
|
|
};
|
|
|
|
let interval = Interval::new(
|
|
Instant::now() + Duration::from_millis(TIMER_DELAY_MS),
|
|
Duration::from_millis(TIMER_INTERVAL_MS),
|
|
);
|
|
|
|
let mut prev_best = match client.best_block_header() {
|
|
Ok(header) => header.hash(),
|
|
Err(e) => {
|
|
warn!("Cant's start consensus service. Error reading best block header: {:?}", e);
|
|
return;
|
|
}
|
|
};
|
|
|
|
let timed = {
|
|
let c = client.clone();
|
|
let s = bft_service.clone();
|
|
|
|
interval.map_err(|e| debug!("Timer error: {:?}", e)).for_each(move |_| {
|
|
if let Ok(best_block) = c.best_block_header() {
|
|
let hash = best_block.hash();
|
|
if hash == prev_best && s.live_agreement() != Some(hash) {
|
|
debug!("Starting consensus round after a timeout");
|
|
start_bft(best_block, s.clone());
|
|
}
|
|
prev_best = hash;
|
|
}
|
|
Ok(())
|
|
})
|
|
};
|
|
|
|
runtime.spawn(notifications);
|
|
runtime.spawn(timed);
|
|
|
|
let prune_available = prune_unneeded_availability(client, extrinsic_store)
|
|
.select(exit.clone())
|
|
.then(|_| Ok(()));
|
|
|
|
// spawn this on the tokio executor since it's fine on a thread pool.
|
|
thread_pool.spawn(prune_available);
|
|
|
|
if let Err(e) = runtime.block_on(exit) {
|
|
debug!("BFT event loop error {:?}", e);
|
|
}
|
|
});
|
|
Service {
|
|
thread: Some(thread),
|
|
exit_signal: Some(signal),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl Drop for Service {
|
|
fn drop(&mut self) {
|
|
if let Some(signal) = self.exit_signal.take() {
|
|
signal.fire();
|
|
}
|
|
|
|
if let Some(thread) = self.thread.take() {
|
|
thread.join().expect("The service thread has panicked");
|
|
}
|
|
}
|
|
}
|