// Copyright 2017-2019 Parity Technologies (UK) Ltd. // This file is part of Substrate. // 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 Substrate. If not, see . use crate::{Service, NetworkStatus, NetworkState, error::Error, DEFAULT_PROTOCOL_ID}; use crate::{SpawnTaskHandle, start_rpc_servers, build_network_future, TransactionPoolAdapter}; use crate::status_sinks; use crate::config::{Configuration, DatabaseConfig}; use client_api::{ self, BlockchainEvents, backend::RemoteBackend, light::RemoteBlockchain, }; use client::Client; use chain_spec::{RuntimeGenesis, Extension}; use consensus_common::import_queue::ImportQueue; use futures::{prelude::*, sync::mpsc}; use futures03::{ compat::{Compat, Future01CompatExt}, FutureExt as _, TryFutureExt as _, StreamExt as _, TryStreamExt as _, future::{select, Either} }; use keystore::{Store as Keystore}; use log::{info, warn, error}; use network::{FinalityProofProvider, OnDemand, NetworkService, NetworkStateInfo, DhtEvent}; use network::{config::BoxFinalityProofRequestBuilder, specialization::NetworkSpecialization}; use parking_lot::{Mutex, RwLock}; use primitives::{Blake2Hasher, H256, Hasher}; use rpc; use sp_api::ConstructRuntimeApi; use sp_runtime::generic::BlockId; use sp_runtime::traits::{ Block as BlockT, ProvideRuntimeApi, NumberFor, Header, SaturatedConversion, }; use sc_executor::{NativeExecutor, NativeExecutionDispatch}; use std::{ io::{Read, Write, Seek}, marker::PhantomData, sync::Arc, time::SystemTime }; use sysinfo::{get_current_pid, ProcessExt, System, SystemExt}; use tel::{telemetry, SUBSTRATE_INFO}; use txpool_api::{TransactionPool, TransactionPoolMaintainer}; use sp_blockchain; use grafana_data_source::{self, record_metrics}; /// Aggregator for the components required to build a service. /// /// # Usage /// /// Call [`ServiceBuilder::new_full`] or [`ServiceBuilder::new_light`], then call the various /// `with_` methods to add the required components that you built yourself: /// /// - [`with_select_chain`](ServiceBuilder::with_select_chain) /// - [`with_import_queue`](ServiceBuilder::with_import_queue) /// - [`with_network_protocol`](ServiceBuilder::with_network_protocol) /// - [`with_finality_proof_provider`](ServiceBuilder::with_finality_proof_provider) /// - [`with_transaction_pool`](ServiceBuilder::with_transaction_pool) /// /// After this is done, call [`build`](ServiceBuilder::build) to construct the service. /// /// The order in which the `with_*` methods are called doesn't matter, as the correct binding of /// generics is done when you call `build`. /// pub struct ServiceBuilder { config: Configuration, pub (crate) client: Arc, backend: Arc, keystore: Arc>, fetcher: Option, select_chain: Option, pub (crate) import_queue: TImpQu, finality_proof_request_builder: Option, finality_proof_provider: Option, network_protocol: TNetP, transaction_pool: Arc, rpc_extensions: TRpc, remote_backend: Option>>, dht_event_tx: Option>, marker: PhantomData<(TBl, TRtApi)>, } /// Full client type. type TFullClient = Client< TFullBackend, TFullCallExecutor, TBl, TRtApi, >; /// Full client backend type. type TFullBackend = client_db::Backend; /// Full client call executor type. type TFullCallExecutor = client::LocalCallExecutor< client_db::Backend, NativeExecutor, >; /// Light client type. type TLightClient = Client< TLightBackend, TLightCallExecutor, TBl, TRtApi, >; /// Light client backend type. type TLightBackend = client::light::backend::Backend< client_db::light::LightStorage, Blake2Hasher, >; /// Light call executor type. type TLightCallExecutor = client::light::call_executor::GenesisCallExecutor< client::light::backend::Backend< client_db::light::LightStorage, Blake2Hasher >, client::LocalCallExecutor< client::light::backend::Backend< client_db::light::LightStorage, Blake2Hasher >, NativeExecutor >, >; impl ServiceBuilder<(), (), TCfg, TGen, TCSExt, (), (), (), (), (), (), (), (), (), ()> where TGen: RuntimeGenesis, TCSExt: Extension { /// Start the service builder with a configuration. pub fn new_full, TRtApi, TExecDisp: NativeExecutionDispatch>( config: Configuration ) -> Result, Arc>, (), (), BoxFinalityProofRequestBuilder, Arc>, (), (), (), TFullBackend, >, Error> { let keystore = Keystore::open( config.keystore_path.clone().ok_or("No basepath configured")?, config.keystore_password.clone() )?; let executor = NativeExecutor::::new( config.wasm_method, config.default_heap_pages, ); let fork_blocks = config.chain_spec .extensions() .get::>() .cloned() .unwrap_or_default(); let (client, backend) = { let db_config = client_db::DatabaseSettings { state_cache_size: config.state_cache_size, state_cache_child_ratio: config.state_cache_child_ratio.map(|v| (v, 100)), pruning: config.pruning.clone(), source: match &config.database { DatabaseConfig::Path { path, cache_size } => client_db::DatabaseSettingsSrc::Path { path: path.clone(), cache_size: cache_size.clone().map(|u| u as usize), }, DatabaseConfig::Custom(db) => client_db::DatabaseSettingsSrc::Custom(db.clone()), }, }; let extensions = client_api::execution_extensions::ExecutionExtensions::new( config.execution_strategies.clone(), Some(keystore.clone()), ); client_db::new_client( db_config, executor, &config.chain_spec, fork_blocks, extensions, )? }; let client = Arc::new(client); Ok(ServiceBuilder { config, client, backend, keystore, fetcher: None, select_chain: None, import_queue: (), finality_proof_request_builder: None, finality_proof_provider: None, network_protocol: (), transaction_pool: Arc::new(()), rpc_extensions: Default::default(), remote_backend: None, dht_event_tx: None, marker: PhantomData, }) } /// Start the service builder with a configuration. pub fn new_light, TRtApi, TExecDisp: NativeExecutionDispatch + 'static>( config: Configuration ) -> Result, Arc>, (), (), BoxFinalityProofRequestBuilder, Arc>, (), (), (), TLightBackend, >, Error> { let keystore = Keystore::open( config.keystore_path.clone().ok_or("No basepath configured")?, config.keystore_password.clone() )?; let executor = NativeExecutor::::new( config.wasm_method, config.default_heap_pages, ); let db_storage = { let db_settings = client_db::DatabaseSettings { state_cache_size: config.state_cache_size, state_cache_child_ratio: config.state_cache_child_ratio.map(|v| (v, 100)), pruning: config.pruning.clone(), source: match &config.database { DatabaseConfig::Path { path, cache_size } => client_db::DatabaseSettingsSrc::Path { path: path.clone(), cache_size: cache_size.clone().map(|u| u as usize), }, DatabaseConfig::Custom(db) => client_db::DatabaseSettingsSrc::Custom(db.clone()), }, }; client_db::light::LightStorage::new(db_settings)? }; let light_blockchain = client::light::new_light_blockchain(db_storage); let fetch_checker = Arc::new(client::light::new_fetch_checker(light_blockchain.clone(), executor.clone())); let fetcher = Arc::new(network::OnDemand::new(fetch_checker)); let backend = client::light::new_light_backend(light_blockchain); let remote_blockchain = backend.remote_blockchain(); let client = Arc::new(client::light::new_light( backend.clone(), &config.chain_spec, executor, )?); Ok(ServiceBuilder { config, client, backend, keystore, fetcher: Some(fetcher.clone()), select_chain: None, import_queue: (), finality_proof_request_builder: None, finality_proof_provider: None, network_protocol: (), transaction_pool: Arc::new(()), rpc_extensions: Default::default(), remote_backend: Some(remote_blockchain), dht_event_tx: None, marker: PhantomData, }) } } impl ServiceBuilder { /// Returns a reference to the client that was stored in this builder. pub fn client(&self) -> &Arc { &self.client } /// Returns a reference to the backend that was used in this builder. pub fn backend(&self) -> &Arc { &self.backend } /// Returns a reference to the select-chain that was stored in this builder. pub fn select_chain(&self) -> Option<&TSc> { self.select_chain.as_ref() } /// Defines which head-of-chain strategy to use. pub fn with_opt_select_chain( self, select_chain_builder: impl FnOnce( &Configuration, &Arc ) -> Result, Error> ) -> Result, Error> { let select_chain = select_chain_builder(&self.config, &self.backend)?; Ok(ServiceBuilder { config: self.config, client: self.client, backend: self.backend, keystore: self.keystore, fetcher: self.fetcher, select_chain, import_queue: self.import_queue, finality_proof_request_builder: self.finality_proof_request_builder, finality_proof_provider: self.finality_proof_provider, network_protocol: self.network_protocol, transaction_pool: self.transaction_pool, rpc_extensions: self.rpc_extensions, remote_backend: self.remote_backend, dht_event_tx: self.dht_event_tx, marker: self.marker, }) } /// Defines which head-of-chain strategy to use. pub fn with_select_chain( self, builder: impl FnOnce(&Configuration, &Arc) -> Result ) -> Result, Error> { self.with_opt_select_chain(|cfg, b| builder(cfg, b).map(Option::Some)) } /// Defines which import queue to use. pub fn with_import_queue( self, builder: impl FnOnce(&Configuration, Arc, Option, Arc) -> Result ) -> Result, Error> where TSc: Clone { let import_queue = builder( &self.config, self.client.clone(), self.select_chain.clone(), self.transaction_pool.clone() )?; Ok(ServiceBuilder { config: self.config, client: self.client, backend: self.backend, keystore: self.keystore, fetcher: self.fetcher, select_chain: self.select_chain, import_queue, finality_proof_request_builder: self.finality_proof_request_builder, finality_proof_provider: self.finality_proof_provider, network_protocol: self.network_protocol, transaction_pool: self.transaction_pool, rpc_extensions: self.rpc_extensions, remote_backend: self.remote_backend, dht_event_tx: self.dht_event_tx, marker: self.marker, }) } /// Defines which network specialization protocol to use. pub fn with_network_protocol( self, network_protocol_builder: impl FnOnce(&Configuration) -> Result ) -> Result, Error> { let network_protocol = network_protocol_builder(&self.config)?; Ok(ServiceBuilder { config: self.config, client: self.client, backend: self.backend, keystore: self.keystore, fetcher: self.fetcher, select_chain: self.select_chain, import_queue: self.import_queue, finality_proof_request_builder: self.finality_proof_request_builder, finality_proof_provider: self.finality_proof_provider, network_protocol, transaction_pool: self.transaction_pool, rpc_extensions: self.rpc_extensions, remote_backend: self.remote_backend, dht_event_tx: self.dht_event_tx, marker: self.marker, }) } /// Defines which strategy to use for providing finality proofs. pub fn with_opt_finality_proof_provider( self, builder: impl FnOnce(Arc, Arc) -> Result>>, Error> ) -> Result>, TNetP, TExPool, TRpc, Backend, >, Error> { let finality_proof_provider = builder(self.client.clone(), self.backend.clone())?; Ok(ServiceBuilder { config: self.config, client: self.client, backend: self.backend, keystore: self.keystore, fetcher: self.fetcher, select_chain: self.select_chain, import_queue: self.import_queue, finality_proof_request_builder: self.finality_proof_request_builder, finality_proof_provider, network_protocol: self.network_protocol, transaction_pool: self.transaction_pool, rpc_extensions: self.rpc_extensions, remote_backend: self.remote_backend, dht_event_tx: self.dht_event_tx, marker: self.marker, }) } /// Defines which strategy to use for providing finality proofs. pub fn with_finality_proof_provider( self, build: impl FnOnce(Arc, Arc) -> Result>, Error> ) -> Result>, TNetP, TExPool, TRpc, Backend, >, Error> { self.with_opt_finality_proof_provider(|client, backend| build(client, backend).map(Option::Some)) } /// Defines which import queue to use. pub fn with_import_queue_and_opt_fprb( self, builder: impl FnOnce( &Configuration, Arc, Arc, Option, Option, Arc, ) -> Result<(UImpQu, Option), Error> ) -> Result, Error> where TSc: Clone, TFchr: Clone { let (import_queue, fprb) = builder( &self.config, self.client.clone(), self.backend.clone(), self.fetcher.clone(), self.select_chain.clone(), self.transaction_pool.clone() )?; Ok(ServiceBuilder { config: self.config, client: self.client, backend: self.backend, keystore: self.keystore, fetcher: self.fetcher, select_chain: self.select_chain, import_queue, finality_proof_request_builder: fprb, finality_proof_provider: self.finality_proof_provider, network_protocol: self.network_protocol, transaction_pool: self.transaction_pool, rpc_extensions: self.rpc_extensions, remote_backend: self.remote_backend, dht_event_tx: self.dht_event_tx, marker: self.marker, }) } /// Defines which import queue to use. pub fn with_import_queue_and_fprb( self, builder: impl FnOnce( &Configuration, Arc, Arc, Option, Option, Arc, ) -> Result<(UImpQu, UFprb), Error> ) -> Result, Error> where TSc: Clone, TFchr: Clone { self.with_import_queue_and_opt_fprb(|cfg, cl, b, f, sc, tx| builder(cfg, cl, b, f, sc, tx) .map(|(q, f)| (q, Some(f))) ) } /// Defines which transaction pool to use. pub fn with_transaction_pool( self, transaction_pool_builder: impl FnOnce( txpool::txpool::Options, Arc, Option, ) -> Result ) -> Result, Error> where TSc: Clone, TFchr: Clone { let transaction_pool = transaction_pool_builder( self.config.transaction_pool.clone(), self.client.clone(), self.fetcher.clone(), )?; Ok(ServiceBuilder { config: self.config, client: self.client, backend: self.backend, keystore: self.keystore, fetcher: self.fetcher, select_chain: self.select_chain, import_queue: self.import_queue, finality_proof_request_builder: self.finality_proof_request_builder, finality_proof_provider: self.finality_proof_provider, network_protocol: self.network_protocol, transaction_pool: Arc::new(transaction_pool), rpc_extensions: self.rpc_extensions, remote_backend: self.remote_backend, dht_event_tx: self.dht_event_tx, marker: self.marker, }) } /// Defines the RPC extensions to use. pub fn with_rpc_extensions( self, rpc_ext_builder: impl FnOnce( Arc, Arc, Arc, Option, Option>>, ) -> Result, ) -> Result, Error> where TSc: Clone, TFchr: Clone { let rpc_extensions = rpc_ext_builder( self.client.clone(), self.transaction_pool.clone(), self.backend.clone(), self.fetcher.clone(), self.remote_backend.clone(), )?; Ok(ServiceBuilder { config: self.config, client: self.client, backend: self.backend, keystore: self.keystore, fetcher: self.fetcher, select_chain: self.select_chain, import_queue: self.import_queue, finality_proof_request_builder: self.finality_proof_request_builder, finality_proof_provider: self.finality_proof_provider, network_protocol: self.network_protocol, transaction_pool: self.transaction_pool, rpc_extensions, remote_backend: self.remote_backend, dht_event_tx: self.dht_event_tx, marker: self.marker, }) } /// Adds a dht event sender to builder to be used by the network to send dht events to the authority discovery /// module. pub fn with_dht_event_tx( self, dht_event_tx: mpsc::Sender, ) -> Result, Error> { Ok(ServiceBuilder { config: self.config, client: self.client, backend: self.backend, keystore: self.keystore, fetcher: self.fetcher, select_chain: self.select_chain, import_queue: self.import_queue, finality_proof_request_builder: self.finality_proof_request_builder, finality_proof_provider: self.finality_proof_provider, network_protocol: self.network_protocol, transaction_pool: self.transaction_pool, rpc_extensions: self.rpc_extensions, remote_backend: self.remote_backend, dht_event_tx: Some(dht_event_tx), marker: self.marker, }) } } /// Implemented on `ServiceBuilder`. Allows running block commands, such as import/export/validate /// components to the builder. pub trait ServiceBuilderCommand { /// Block type this API operates on. type Block: BlockT; /// Starts the process of importing blocks. fn import_blocks( self, input: impl Read + Seek + Send + 'static, force: bool, ) -> Box + Send>; /// Performs the blocks export. fn export_blocks( self, output: impl Write + 'static, from: NumberFor, to: Option>, json: bool ) -> Box>; /// Performs a revert of `blocks` blocks. fn revert_chain( &self, blocks: NumberFor ) -> Result<(), Error>; /// Re-validate known block. fn check_block( self, block: BlockId ) -> Box + Send>; } impl ServiceBuilder< TBl, TRtApi, TCfg, TGen, TCSExt, Client, Arc>, TSc, TImpQu, BoxFinalityProofRequestBuilder, Arc>, TNetP, TExPool, TRpc, TBackend, > where Client: ProvideRuntimeApi, as ProvideRuntimeApi>::Api: sp_api::Metadata + offchain::OffchainWorkerApi + txpool_runtime_api::TaggedTransactionQueue + session::SessionKeys + sp_api::ApiExt, TBl: BlockT::Out>, TRtApi: ConstructRuntimeApi> + 'static + Send + Sync, TCfg: Default, TGen: RuntimeGenesis, TCSExt: Extension, TBackend: 'static + client_api::backend::Backend + Send, TExec: 'static + client::CallExecutor + Send + Sync + Clone, TSc: Clone, TImpQu: 'static + ImportQueue, TNetP: NetworkSpecialization, TExPool: 'static + TransactionPool::Hash> + TransactionPoolMaintainer::Hash>, TRpc: rpc::RpcExtension + Clone, { /// Builds the service. pub fn build(self) -> Result, TSc, NetworkStatus, NetworkService::Hash>, TExPool, offchain::OffchainWorkers< Client, TBackend::OffchainStorage, TBl >, >, Error> { let ServiceBuilder { marker: _, mut config, client, fetcher: on_demand, backend, keystore, select_chain, import_queue, finality_proof_request_builder, finality_proof_provider, network_protocol, transaction_pool, rpc_extensions, remote_backend, dht_event_tx, } = self; session::generate_initial_session_keys( client.clone(), &BlockId::Hash(client.info().chain.best_hash), config.dev_key_seed.clone().map(|s| vec![s]).unwrap_or_default(), )?; let (signal, exit) = exit_future::signal(); // List of asynchronous tasks to spawn. We collect them, then spawn them all at once. let (to_spawn_tx, to_spawn_rx) = mpsc::unbounded:: + Send>>(); // A side-channel for essential tasks to communicate shutdown. let (essential_failed_tx, essential_failed_rx) = mpsc::unbounded(); let import_queue = Box::new(import_queue); let chain_info = client.info().chain; let version = config.full_version(); info!("Highest known block at #{}", chain_info.best_number); telemetry!( SUBSTRATE_INFO; "node.start"; "height" => chain_info.best_number.saturated_into::(), "best" => ?chain_info.best_hash ); // make transaction pool available for off-chain runtime calls. client.execution_extensions() .register_transaction_pool(Arc::downgrade(&transaction_pool) as _); let transaction_pool_adapter = Arc::new(TransactionPoolAdapter { imports_external_transactions: !config.roles.is_light(), pool: transaction_pool.clone(), client: client.clone(), executor: Arc::new(SpawnTaskHandle { sender: to_spawn_tx.clone(), on_exit: exit.clone() }), }); let protocol_id = { let protocol_id_full = match config.chain_spec.protocol_id() { Some(pid) => pid, None => { warn!("Using default protocol ID {:?} because none is configured in the \ chain specs", DEFAULT_PROTOCOL_ID ); DEFAULT_PROTOCOL_ID } }.as_bytes(); network::config::ProtocolId::from(protocol_id_full) }; let block_announce_validator = Box::new(consensus_common::block_validation::DefaultBlockAnnounceValidator::new(client.clone())); let network_params = network::config::Params { roles: config.roles, network_config: config.network.clone(), chain: client.clone(), finality_proof_provider, finality_proof_request_builder, on_demand: on_demand.clone(), transaction_pool: transaction_pool_adapter.clone() as _, import_queue, protocol_id, specialization: network_protocol, block_announce_validator, }; let has_bootnodes = !network_params.network_config.boot_nodes.is_empty(); let network_mut = network::NetworkWorker::new(network_params)?; let network = network_mut.service().clone(); let network_status_sinks = Arc::new(Mutex::new(status_sinks::StatusSinks::new())); let offchain_storage = backend.offchain_storage(); let offchain_workers = match (config.offchain_worker, offchain_storage) { (true, Some(db)) => { Some(Arc::new(offchain::OffchainWorkers::new(client.clone(), db))) }, (true, None) => { log::warn!("Offchain workers disabled, due to lack of offchain storage support in backend."); None }, _ => None, }; { // block notifications let txpool = Arc::downgrade(&transaction_pool); let offchain = offchain_workers.as_ref().map(Arc::downgrade); let to_spawn_tx_ = to_spawn_tx.clone(); let network_state_info: Arc = network.clone(); let is_validator = config.roles.is_authority(); let events = client.import_notification_stream() .map(|v| Ok::<_, ()>(v)).compat() .for_each(move |notification| { let number = *notification.header.number(); let txpool = txpool.upgrade(); if let Some(txpool) = txpool.as_ref() { let future = txpool.maintain( &BlockId::hash(notification.hash), ¬ification.retracted, ).map(|_| Ok(())).compat(); let _ = to_spawn_tx_.unbounded_send(Box::new(future)); } let offchain = offchain.as_ref().and_then(|o| o.upgrade()); if let Some(offchain) = offchain { let future = offchain.on_block_imported(&number, network_state_info.clone(), is_validator) .map(|()| Ok(())); let _ = to_spawn_tx_.unbounded_send(Box::new(Compat::new(future))); } Ok(()) }) .select(exit.clone()) .then(|_| Ok(())); let _ = to_spawn_tx.unbounded_send(Box::new(events)); } { // extrinsic notifications let network = Arc::downgrade(&network); let transaction_pool_ = transaction_pool.clone(); let events = transaction_pool.import_notification_stream() .map(|v| Ok::<_, ()>(v)).compat() .for_each(move |_| { if let Some(network) = network.upgrade() { network.trigger_repropagate(); } let status = transaction_pool_.status(); telemetry!(SUBSTRATE_INFO; "txpool.import"; "ready" => status.ready, "future" => status.future ); Ok(()) }) .select(exit.clone()) .then(|_| Ok(())); let _ = to_spawn_tx.unbounded_send(Box::new(events)); } // Periodically notify the telemetry. let transaction_pool_ = transaction_pool.clone(); let client_ = client.clone(); let mut sys = System::new(); let self_pid = get_current_pid().ok(); let (state_tx, state_rx) = mpsc::unbounded::<(NetworkStatus<_>, NetworkState)>(); network_status_sinks.lock().push(std::time::Duration::from_millis(5000), state_tx); let tel_task = state_rx.for_each(move |(net_status, _)| { let info = client_.info(); let best_number = info.chain.best_number.saturated_into::(); let best_hash = info.chain.best_hash; let num_peers = net_status.num_connected_peers; let txpool_status = transaction_pool_.status(); let finalized_number: u64 = info.chain.finalized_number.saturated_into::(); let bandwidth_download = net_status.average_download_per_sec; let bandwidth_upload = net_status.average_upload_per_sec; let used_state_cache_size = match info.used_state_cache_size { Some(size) => size, None => 0, }; // get cpu usage and memory usage of this process let (cpu_usage, memory) = if let Some(self_pid) = self_pid { if sys.refresh_process(self_pid) { let proc = sys.get_process(self_pid) .expect("Above refresh_process succeeds, this should be Some(), qed"); (proc.cpu_usage(), proc.memory()) } else { (0.0, 0) } } else { (0.0, 0) }; telemetry!( SUBSTRATE_INFO; "system.interval"; "peers" => num_peers, "height" => best_number, "best" => ?best_hash, "txcount" => txpool_status.ready, "cpu" => cpu_usage, "memory" => memory, "finalized_height" => finalized_number, "finalized_hash" => ?info.chain.finalized_hash, "bandwidth_download" => bandwidth_download, "bandwidth_upload" => bandwidth_upload, "used_state_cache_size" => used_state_cache_size, ); record_metrics!( "peers".to_owned() => num_peers, "height".to_owned() => best_number, "txcount".to_owned() => txpool_status.ready, "cpu".to_owned() => cpu_usage, "memory".to_owned() => memory, "finalized_height".to_owned() => finalized_number, "bandwidth_download".to_owned() => bandwidth_download, "bandwidth_upload".to_owned() => bandwidth_upload, "used_state_cache_size".to_owned() => used_state_cache_size ); Ok(()) }).select(exit.clone()).then(|_| Ok(())); let _ = to_spawn_tx.unbounded_send(Box::new(tel_task)); // Periodically send the network state to the telemetry. let (netstat_tx, netstat_rx) = mpsc::unbounded::<(NetworkStatus<_>, NetworkState)>(); network_status_sinks.lock().push(std::time::Duration::from_secs(30), netstat_tx); let tel_task_2 = netstat_rx.for_each(move |(_, network_state)| { telemetry!( SUBSTRATE_INFO; "system.network_state"; "state" => network_state, ); Ok(()) }).select(exit.clone()).then(|_| Ok(())); let _ = to_spawn_tx.unbounded_send(Box::new(tel_task_2)); // RPC let (system_rpc_tx, system_rpc_rx) = futures03::channel::mpsc::unbounded(); let gen_handler = || { use rpc::{chain, state, author, system}; let system_info = rpc::system::SystemInfo { chain_name: config.chain_spec.name().into(), impl_name: config.impl_name.into(), impl_version: config.impl_version.into(), properties: config.chain_spec.properties().clone(), }; let subscriptions = rpc::Subscriptions::new(Arc::new(SpawnTaskHandle { sender: to_spawn_tx.clone(), on_exit: exit.clone() })); let (chain, state) = if let (Some(remote_backend), Some(on_demand)) = (remote_backend.as_ref(), on_demand.as_ref()) { // Light clients let chain = rpc::chain::new_light( client.clone(), subscriptions.clone(), remote_backend.clone(), on_demand.clone() ); let state = rpc::state::new_light( client.clone(), subscriptions.clone(), remote_backend.clone(), on_demand.clone() ); (chain, state) } else { // Full nodes let chain = rpc::chain::new_full(client.clone(), subscriptions.clone()); let state = rpc::state::new_full(client.clone(), subscriptions.clone()); (chain, state) }; let author = rpc::author::Author::new( client.clone(), transaction_pool.clone(), subscriptions, keystore.clone(), ); let system = system::System::new(system_info, system_rpc_tx.clone()); rpc_servers::rpc_handler(( state::StateApi::to_delegate(state), chain::ChainApi::to_delegate(chain), author::AuthorApi::to_delegate(author), system::SystemApi::to_delegate(system), rpc_extensions.clone(), )) }; let rpc_handlers = gen_handler(); let rpc = start_rpc_servers(&config, gen_handler)?; let _ = to_spawn_tx.unbounded_send(Box::new(build_network_future( config.roles, network_mut, client.clone(), network_status_sinks.clone(), system_rpc_rx, has_bootnodes, dht_event_tx, ) .map_err(|_| ()) .select(exit.clone()) .then(|_| Ok(())))); let telemetry_connection_sinks: Arc>>> = Default::default(); // Telemetry let telemetry = config.telemetry_endpoints.clone().map(|endpoints| { let is_authority = config.roles.is_authority(); let network_id = network.local_peer_id().to_base58(); let name = config.name.clone(); let impl_name = config.impl_name.to_owned(); let version = version.clone(); let chain_name = config.chain_spec.name().to_owned(); let telemetry_connection_sinks_ = telemetry_connection_sinks.clone(); let telemetry = tel::init_telemetry(tel::TelemetryConfig { endpoints, wasm_external_transport: config.telemetry_external_transport.take(), }); let startup_time = SystemTime::UNIX_EPOCH.elapsed() .map(|dur| dur.as_millis()) .unwrap_or(0); let future = telemetry.clone() .map(|ev| Ok::<_, ()>(ev)) .compat() .for_each(move |event| { // Safe-guard in case we add more events in the future. let tel::TelemetryEvent::Connected = event; telemetry!(SUBSTRATE_INFO; "system.connected"; "name" => name.clone(), "implementation" => impl_name.clone(), "version" => version.clone(), "config" => "", "chain" => chain_name.clone(), "authority" => is_authority, "startup_time" => startup_time, "network_id" => network_id.clone() ); telemetry_connection_sinks_.lock().retain(|sink| { sink.unbounded_send(()).is_ok() }); Ok(()) }); let _ = to_spawn_tx.unbounded_send(Box::new(future .select(exit.clone()) .then(|_| Ok(())))); telemetry }); // Grafana data source if let Some(port) = config.grafana_port { let future = select( grafana_data_source::run_server(port).boxed(), exit.clone().compat() ).map(|either| match either { Either::Left((result, _)) => result.map_err(|_| ()), Either::Right(_) => Ok(()) }).compat(); let _ = to_spawn_tx.unbounded_send(Box::new(future)); } // Instrumentation if let Some(tracing_targets) = config.tracing_targets.as_ref() { let subscriber = sc_tracing::ProfilingSubscriber::new( config.tracing_receiver, tracing_targets ); match tracing::subscriber::set_global_default(subscriber) { Ok(_) => (), Err(e) => error!(target: "tracing", "Unable to set global default subscriber {}", e), } } Ok(Service { client, network, network_status_sinks, select_chain, transaction_pool, exit, signal: Some(signal), essential_failed_tx, essential_failed_rx, to_spawn_tx, to_spawn_rx, to_poll: Vec::new(), rpc_handlers, _rpc: rpc, _telemetry: telemetry, _offchain_workers: offchain_workers, _telemetry_on_connect_sinks: telemetry_connection_sinks.clone(), keystore, marker: PhantomData::, }) } }