Skip to content
Snippets Groups Projects
service.rs 8.46 KiB
Newer Older
Shawn Tabrizi's avatar
Shawn Tabrizi committed
//! Service and ServiceFactory implementation. Specialized wrapper over substrate service.

use std::sync::Arc;
use std::time::Duration;
Joshy Orndorff's avatar
Joshy Orndorff committed
use sc_client::LongestChain;
use node_template_runtime::{self, GenesisConfig, opaque::Block, RuntimeApi};
Joshy Orndorff's avatar
Joshy Orndorff committed
use sc_service::{error::{Error as ServiceError}, AbstractService, Configuration, ServiceBuilder};
use sp_inherents::InherentDataProviders;
use sc_network::{construct_simple_protocol};
use sc_executor::native_executor_instance;
pub use sc_executor::NativeExecutor;
use sp_consensus_aura::sr25519::{AuthorityPair as AuraPair};
Shawn Tabrizi's avatar
Shawn Tabrizi committed
use grandpa::{self, FinalityProofProvider as GrandpaFinalityProofProvider};
Joshy Orndorff's avatar
Joshy Orndorff committed
use sc_basic_authority;
Shawn Tabrizi's avatar
Shawn Tabrizi committed

// Our native executor instance.
native_executor_instance!(
	pub Executor,
	node_template_runtime::api::dispatch,
	node_template_runtime::native_version,
Shawn Tabrizi's avatar
Shawn Tabrizi committed
);

construct_simple_protocol! {
	/// Demo protocol attachment for substrate.
	pub struct NodeProtocol where Block = Block { }
}

/// Starts a `ServiceBuilder` for a full service.
///
/// Use this macro if you don't actually need the full service, but just the builder in order to
/// be able to perform chain operations.
macro_rules! new_full_start {
	($config:expr) => {{
		let mut import_setup = None;
Joshy Orndorff's avatar
Joshy Orndorff committed
		let inherent_data_providers = sp_inherents::InherentDataProviders::new();
Joshy Orndorff's avatar
Joshy Orndorff committed
		let builder = sc_service::ServiceBuilder::new_full::<
Shawn Tabrizi's avatar
Shawn Tabrizi committed
			node_template_runtime::opaque::Block, node_template_runtime::RuntimeApi, crate::service::Executor
		>($config)?
			.with_select_chain(|_config, backend| {
Joshy Orndorff's avatar
Joshy Orndorff committed
				Ok(sc_client::LongestChain::new(backend.clone()))
			})?
			.with_transaction_pool(|config, client, _fetcher| {
				let pool_api = sc_transaction_pool::FullChainApi::new(client.clone());
				let pool = sc_transaction_pool::BasicPool::new(config, pool_api);
				let maintainer = sc_transaction_pool::FullBasicPoolMaintainer::new(pool.pool().clone(), client);
				let maintainable_pool = sp_transaction_pool::MaintainableTransactionPool::new(pool, maintainer);
				Ok(maintainable_pool)
Shawn Tabrizi's avatar
Shawn Tabrizi committed
			})?
Shawn Tabrizi's avatar
Shawn Tabrizi committed
			.with_import_queue(|_config, client, mut select_chain, transaction_pool| {
Shawn Tabrizi's avatar
Shawn Tabrizi committed
				let select_chain = select_chain.take()
Joshy Orndorff's avatar
Joshy Orndorff committed
					.ok_or_else(|| sc_service::Error::SelectChainRequired)?;
				let (grandpa_block_import, grandpa_link) =
Joshy Orndorff's avatar
Joshy Orndorff committed
					grandpa::block_import::<_, _, _, node_template_runtime::RuntimeApi, _>(
						client.clone(), &*client, select_chain
Shawn Tabrizi's avatar
Shawn Tabrizi committed
					)?;
Ricardo Rius's avatar
Ricardo Rius committed
				let aura_block_import = sc_consensus_aura::AuraBlockImport::<_, _, _, AuraPair>::new(
					grandpa_block_import.clone(), client.clone(),
				);

				let import_queue = sc_consensus_aura::import_queue::<_, _, _, AuraPair, _>(
Joshy Orndorff's avatar
Joshy Orndorff committed
					sc_consensus_aura::SlotDuration::get_or_compute(&*client)?,
Ricardo Rius's avatar
Ricardo Rius committed
					aura_block_import,
Shawn Tabrizi's avatar
Shawn Tabrizi committed
					Some(Box::new(grandpa_block_import.clone())),
Shawn Tabrizi's avatar
Shawn Tabrizi committed
					None,
					client,
					inherent_data_providers.clone(),
Shawn Tabrizi's avatar
Shawn Tabrizi committed
					Some(transaction_pool),
Shawn Tabrizi's avatar
Shawn Tabrizi committed
				import_setup = Some((grandpa_block_import, grandpa_link));
Shawn Tabrizi's avatar
Shawn Tabrizi committed

				Ok(import_queue)
			})?;

		(builder, import_setup, inherent_data_providers)
Shawn Tabrizi's avatar
Shawn Tabrizi committed
	}}
}

/// Builds a new service for a full client.
pub fn new_full<C: Send + Default + 'static>(config: Configuration<C, GenesisConfig>)
	-> Result<impl AbstractService, ServiceError>
{
	let is_authority = config.roles.is_authority();
Shawn Tabrizi's avatar
Shawn Tabrizi committed
	let force_authoring = config.force_authoring;
	let name = config.name.clone();
	let disable_grandpa = config.disable_grandpa;

Joshy Orndorff's avatar
Joshy Orndorff committed
	// sentry nodes announce themselves as authorities to the network
	// and should run the same protocols authorities do, but it should
	// never actively participate in any consensus process.
	let participates_in_consensus = is_authority && !config.sentry_mode;

	let (builder, mut import_setup, inherent_data_providers) = new_full_start!(config);
Shawn Tabrizi's avatar
Shawn Tabrizi committed
	let (block_import, grandpa_link) =
		import_setup.take()
			.expect("Link Half and Block Import are present for Full Services or setup failed before. qed");

Shawn Tabrizi's avatar
Shawn Tabrizi committed
	let service = builder.with_network_protocol(|_| Ok(NodeProtocol::new()))?
		.with_finality_proof_provider(|client, backend|
			Ok(Arc::new(GrandpaFinalityProofProvider::new(backend, client)) as _)
Shawn Tabrizi's avatar
Shawn Tabrizi committed
		)?
		.build()?;

Joshy Orndorff's avatar
Joshy Orndorff committed
	if participates_in_consensus {
		let proposer = sc_basic_authority::ProposerFactory {
Shawn Tabrizi's avatar
Shawn Tabrizi committed
			client: service.client(),
			transaction_pool: service.transaction_pool(),
		};

		let client = service.client();
		let select_chain = service.select_chain()
			.ok_or(ServiceError::SelectChainRequired)?;

Joshy Orndorff's avatar
Joshy Orndorff committed
		let can_author_with =
			sp_consensus::CanAuthorWithNativeVersion::new(client.executor().clone());

		let aura = sc_consensus_aura::start_aura::<_, _, _, _, _, AuraPair, _, _, _, _>(
			sc_consensus_aura::SlotDuration::get_or_compute(&*client)?,
Shawn Tabrizi's avatar
Shawn Tabrizi committed
			client,
			select_chain,
Shawn Tabrizi's avatar
Shawn Tabrizi committed
			proposer,
			service.network(),
			inherent_data_providers.clone(),
Shawn Tabrizi's avatar
Shawn Tabrizi committed
			service.keystore(),
Joshy Orndorff's avatar
Joshy Orndorff committed
			can_author_with,
Shawn Tabrizi's avatar
Shawn Tabrizi committed
		)?;
Shawn Tabrizi's avatar
Shawn Tabrizi committed
		// the AURA authoring task is considered essential, i.e. if it
Shawn Tabrizi's avatar
Shawn Tabrizi committed
		// fails we take down the service with it.
Joshy Orndorff's avatar
Joshy Orndorff committed
		service.spawn_essential_task(aura);
Joshy Orndorff's avatar
Joshy Orndorff committed
	// if the node isn't actively participating in consensus then it doesn't
	// need a keystore, regardless of which protocol we use below.
	let keystore = if participates_in_consensus {
		Some(service.keystore())
	} else {
		None
	};

	let grandpa_config = grandpa::Config {
Shawn Tabrizi's avatar
Shawn Tabrizi committed
		// FIXME #1578 make this available through chainspec
		gossip_duration: Duration::from_millis(333),
		justification_period: 512,
		name: Some(name),
Joshy Orndorff's avatar
Joshy Orndorff committed
		observer_enabled: true,
		keystore,
		is_authority,
	match (is_authority, disable_grandpa) {
Shawn Tabrizi's avatar
Shawn Tabrizi committed
		(false, false) => {
			// start the lightweight GRANDPA observer
Joshy Orndorff's avatar
Joshy Orndorff committed
			service.spawn_task(grandpa::run_grandpa_observer(
				grandpa_config,
				grandpa_link,
Shawn Tabrizi's avatar
Shawn Tabrizi committed
				service.network(),
				service.on_exit(),
Joshy Orndorff's avatar
Joshy Orndorff committed
				service.spawn_task_handle(),
			)?);
Shawn Tabrizi's avatar
Shawn Tabrizi committed
		},
		(true, false) => {
			// start the full GRANDPA voter
			let voter_config = grandpa::GrandpaParams {
				config: grandpa_config,
				link: grandpa_link,
Shawn Tabrizi's avatar
Shawn Tabrizi committed
				network: service.network(),
				inherent_data_providers: inherent_data_providers.clone(),
				on_exit: service.on_exit(),
				telemetry_on_connect: Some(service.telemetry_on_connect_stream()),
Shawn Tabrizi's avatar
Shawn Tabrizi committed
				voting_rule: grandpa::VotingRulesBuilder::default().build(),
Joshy Orndorff's avatar
Joshy Orndorff committed
				executor: service.spawn_task_handle(),
Shawn Tabrizi's avatar
Shawn Tabrizi committed
			};

			// the GRANDPA voter task is considered infallible, i.e.
			// if it fails we take down the service with it.
			service.spawn_essential_task(grandpa::run_grandpa_voter(voter_config)?);
Shawn Tabrizi's avatar
Shawn Tabrizi committed
		},
		(_, true) => {
			grandpa::setup_disabled_grandpa(
				service.client(),
				&inherent_data_providers,
				service.network(),
			)?;
		},
	}

	Ok(service)
}

/// Builds a new service for a light client.
pub fn new_light<C: Send + Default + 'static>(config: Configuration<C, GenesisConfig>)
	-> Result<impl AbstractService, ServiceError>
{
	let inherent_data_providers = InherentDataProviders::new();

	ServiceBuilder::new_light::<Block, RuntimeApi, Executor>(config)?
		.with_select_chain(|_config, backend| {
			Ok(LongestChain::new(backend.clone()))
Shawn Tabrizi's avatar
Shawn Tabrizi committed
		})?
Joshy Orndorff's avatar
Joshy Orndorff committed
		.with_transaction_pool(|config, client, fetcher| {
			let fetcher = fetcher
				.ok_or_else(|| "Trying to start light transaction pool without active fetcher")?;
			let pool_api = sc_transaction_pool::LightChainApi::new(client.clone(), fetcher.clone());
			let pool = sc_transaction_pool::BasicPool::new(config, pool_api);
			let maintainer = sc_transaction_pool::LightBasicPoolMaintainer::with_defaults(pool.pool().clone(), client, fetcher);
			let maintainable_pool = sp_transaction_pool::MaintainableTransactionPool::new(pool, maintainer);
			Ok(maintainable_pool)
		})?
		.with_import_queue_and_fprb(|_config, client, backend, fetcher, _select_chain, _tx_pool| {
			let fetch_checker = fetcher
Shawn Tabrizi's avatar
Shawn Tabrizi committed
				.map(|fetcher| fetcher.checker().clone())
				.ok_or_else(|| "Trying to start light import queue without active fetch checker")?;
Joshy Orndorff's avatar
Joshy Orndorff committed
			let grandpa_block_import = grandpa::light_block_import::<_, _, _, RuntimeApi>(
				client.clone(), backend, &*client.clone(), Arc::new(fetch_checker),
Shawn Tabrizi's avatar
Shawn Tabrizi committed
			)?;
			let finality_proof_import = grandpa_block_import.clone();
Shawn Tabrizi's avatar
Shawn Tabrizi committed
			let finality_proof_request_builder =
				finality_proof_import.create_finality_proof_request_builder();

Ricardo Rius's avatar
Ricardo Rius committed
			let import_queue = sc_consensus_aura::import_queue::<_, _, _, AuraPair, ()>(
Joshy Orndorff's avatar
Joshy Orndorff committed
				sc_consensus_aura::SlotDuration::get_or_compute(&*client)?,
Ricardo Rius's avatar
Ricardo Rius committed
				grandpa_block_import,
Shawn Tabrizi's avatar
Shawn Tabrizi committed
				None,
				Some(Box::new(finality_proof_import)),
				client,
				inherent_data_providers.clone(),
Shawn Tabrizi's avatar
Shawn Tabrizi committed
				None,
Shawn Tabrizi's avatar
Shawn Tabrizi committed
			)?;

			Ok((import_queue, finality_proof_request_builder))
		})?
		.with_network_protocol(|_| Ok(NodeProtocol::new()))?
		.with_finality_proof_provider(|client, backend|
			Ok(Arc::new(GrandpaFinalityProofProvider::new(backend, client)) as _)
Shawn Tabrizi's avatar
Shawn Tabrizi committed
		)?
		.build()
}