// Copyright 2020 Parity Technologies (UK) Ltd. // This file is part of Cumulus. // Substrate 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. // Substrate 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 Cumulus. If not, see . //! Cumulus service //! //! Provides functions for starting a collator node or a normal full node. use cumulus_primitives::ParaId; use futures::{Future, FutureExt}; use polkadot_overseer::OverseerHandler; use polkadot_primitives::v1::{Block as PBlock, CollatorId, CollatorPair}; use polkadot_service::{AbstractClient, Client as PClient, ClientHandle, RuntimeApiCollection}; use sc_client_api::{Backend as BackendT, BlockBackend, Finalizer, UsageProvider}; use sc_service::{error::Result as ServiceResult, Configuration, Role, TaskManager}; use sp_blockchain::HeaderBackend; use sp_consensus::{BlockImport, Environment, Error as ConsensusError, Proposer}; use sp_core::traits::SpawnNamed; use sp_inherents::InherentDataProviders; use sp_runtime::traits::{BlakeTwo256, Block as BlockT}; use std::{marker::PhantomData, sync::Arc}; /// Polkadot full node handles. type PFullNode = polkadot_service::NewFull; /// Parameters given to [`start_collator`]. pub struct StartCollatorParams<'a, Block: BlockT, PF, BI, BS, Client, Backend, Spawner, PClient> { pub proposer_factory: PF, pub inherent_data_providers: InherentDataProviders, pub backend: Arc, pub block_import: BI, pub block_status: Arc, pub client: Arc, pub announce_block: Arc) + Send + Sync>, pub spawner: Spawner, pub para_id: ParaId, pub collator_key: CollatorPair, pub polkadot_full_node: PFullNode, pub task_manager: &'a mut TaskManager, } /// Start a collator node for a parachain. /// /// A collator is similar to a validator in a normal blockchain. /// It is responsible for producing blocks and sending the blocks to a /// parachain validator for validation and inclusion into the relay chain. pub async fn start_collator<'a, Block, PF, BI, BS, Client, Backend, Spawner, PClient>( StartCollatorParams { proposer_factory, inherent_data_providers, backend, block_import, block_status, client, announce_block, spawner, para_id, collator_key, polkadot_full_node, task_manager, }: StartCollatorParams<'a, Block, PF, BI, BS, Client, Backend, Spawner, PClient>, ) -> sc_service::error::Result<()> where Block: BlockT, PF: Environment + Send + 'static, BI: BlockImport< Block, Error = ConsensusError, Transaction = >::Transaction, > + Send + Sync + 'static, BS: BlockBackend + Send + Sync + 'static, Client: Finalizer + UsageProvider + HeaderBackend + Send + Sync + BlockBackend + 'static, for<'b> &'b Client: BlockImport, Backend: BackendT + 'static, Spawner: SpawnNamed + Clone + Send + Sync + 'static, PClient: ClientHandle, { polkadot_full_node .client .execute_with(StartCollator { proposer_factory, inherent_data_providers, backend, client, announce_block, overseer_handler: polkadot_full_node .overseer_handler .ok_or_else(|| "Polkadot full node did not provided an `OverseerHandler`!")?, spawner, para_id, collator_key, block_import, block_status, }) .await?; task_manager.add_child(polkadot_full_node.task_manager); Ok(()) } struct StartCollator { proposer_factory: PF, inherent_data_providers: InherentDataProviders, backend: Arc, block_import: BI, block_status: Arc, client: Arc, announce_block: Arc) + Send + Sync>, overseer_handler: OverseerHandler, spawner: Spawner, para_id: ParaId, collator_key: CollatorPair, } impl polkadot_service::ExecuteWithClient for StartCollator where Block: BlockT, PF: Environment + Send + 'static, BI: BlockImport< Block, Error = ConsensusError, Transaction = >::Transaction, > + Send + Sync + 'static, BS: BlockBackend + Send + Sync + 'static, Client: Finalizer + UsageProvider + HeaderBackend + Send + Sync + BlockBackend + 'static, for<'b> &'b Client: BlockImport, Backend: BackendT + 'static, Spawner: SpawnNamed + Clone + Send + Sync + 'static, { type Output = std::pin::Pin>>>; fn execute_with_client(self, client: Arc) -> Self::Output where >::StateBackend: sp_api::StateBackend, PBackend: sc_client_api::Backend + 'static, PBackend::State: sp_api::StateBackend, Api: RuntimeApiCollection, PClient: AbstractClient + 'static, { async move { cumulus_collator::start_collator(cumulus_collator::StartCollatorParams { proposer_factory: self.proposer_factory, inherent_data_providers: self.inherent_data_providers, backend: self.backend, block_import: self.block_import, block_status: self.block_status, client: self.client, announce_block: self.announce_block, overseer_handler: self.overseer_handler, spawner: self.spawner, para_id: self.para_id, key: self.collator_key, polkadot_client: client, }) .await .map_err(Into::into) } .boxed() } } /// Parameters given to [`start_full_node`]. pub struct StartFullNodeParams<'a, Block: BlockT, Client, PClient> { pub para_id: ParaId, pub client: Arc, pub polkadot_full_node: PFullNode, pub task_manager: &'a mut TaskManager, pub announce_block: Arc) + Send + Sync>, } /// Start a full node for a parachain. /// /// A full node will only sync the given parachain and will follow the /// tip of the chain. pub fn start_full_node( StartFullNodeParams { client, announce_block, task_manager, polkadot_full_node, para_id, }: StartFullNodeParams, ) -> sc_service::error::Result<()> where Block: BlockT, Client: Finalizer + UsageProvider + Send + Sync + BlockBackend + 'static, for<'a> &'a Client: BlockImport, Backend: BackendT + 'static, PClient: ClientHandle, { polkadot_full_node.client.execute_with(StartFullNode { announce_block, para_id, client, task_manager, _phantom: PhantomData, })?; task_manager.add_child(polkadot_full_node.task_manager); Ok(()) } struct StartFullNode<'a, Block: BlockT, Client, Backend> { para_id: ParaId, announce_block: Arc) + Send + Sync>, client: Arc, task_manager: &'a mut TaskManager, _phantom: PhantomData, } impl<'a, Block, Client, Backend> polkadot_service::ExecuteWithClient for StartFullNode<'a, Block, Client, Backend> where Block: BlockT, Client: Finalizer + UsageProvider + Send + Sync + BlockBackend + 'static, for<'b> &'b Client: BlockImport, Backend: BackendT + 'static, { type Output = ServiceResult<()>; fn execute_with_client(self, client: Arc) -> Self::Output where >::StateBackend: sp_api::StateBackend, PBackend: sc_client_api::Backend, PBackend::State: sp_api::StateBackend, Api: RuntimeApiCollection, PClient: AbstractClient + 'static, { let future = cumulus_consensus::follow_polkadot( self.para_id, self.client, client, self.announce_block, )?; self.task_manager .spawn_essential_handle() .spawn("cumulus-consensus", future); Ok(()) } } /// Prepare the parachain's node condifugration /// /// This function will disable the default announcement of Substrate for the parachain in favor /// of the one of Cumulus. pub fn prepare_node_config(mut parachain_config: Configuration) -> Configuration { parachain_config.announce_block = false; parachain_config } /// Build the Polkadot full node using the given `config`. #[sc_cli::prefix_logs_with("Relaychain")] pub fn build_polkadot_full_node( config: Configuration, collator_id: CollatorId, ) -> sc_service::error::Result> { let is_light = matches!(config.role, Role::Light); if is_light { Err("Light client not supported.".into()) } else { polkadot_service::build_full( config, polkadot_service::IsCollator::Yes(collator_id), None, ) } }