// Copyright 2018-2020 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 . pub use sc_cli::VersionInfo; use tokio::runtime::{Builder as RuntimeBuilder, Runtime}; use sc_cli::{IntoExit, NoCustom, SharedParams, ImportParams, error}; use sc_service::{AbstractService, Roles as ServiceRoles, Configuration}; use log::info; use structopt::StructOpt; use sc_cli::{display_role, parse_and_prepare, GetSharedParams, ParseAndPrepare}; use crate::{service, ChainSpec, load_spec}; use crate::factory_impl::FactoryState; use node_transaction_factory::RuntimeAdapter; use futures::{channel::oneshot, future::{select, Either}}; /// Custom subcommands. #[derive(Clone, Debug, StructOpt)] pub enum CustomSubcommands { /// The custom factory subcommmand for manufacturing transactions. #[structopt( name = "factory", about = "Manufactures num transactions from Alice to random accounts. \ Only supported for development or local testnet." )] Factory(FactoryCmd), } impl GetSharedParams for CustomSubcommands { fn shared_params(&self) -> Option<&SharedParams> { match self { CustomSubcommands::Factory(cmd) => Some(&cmd.shared_params), } } } /// The `factory` command used to generate transactions. /// Please note: this command currently only works on an empty database! #[derive(Debug, StructOpt, Clone)] pub struct FactoryCmd { /// How often to repeat. This option only has an effect in mode `MasterToNToM`. #[structopt(long="rounds", default_value = "1")] pub rounds: u64, /// MasterToN: Manufacture `num` transactions from the master account /// to `num` randomly created accounts, one each. /// /// MasterTo1: Manufacture `num` transactions from the master account /// to exactly one other randomly created account. /// /// MasterToNToM: Manufacture `num` transactions from the master account /// to `num` randomly created accounts. /// From each of these randomly created accounts manufacture /// a transaction to another randomly created account. /// Repeat this `rounds` times. If `rounds` = 1 the behavior /// is the same as `MasterToN`.{n} /// A -> B, A -> C, A -> D, ... x `num`{n} /// B -> E, C -> F, D -> G, ...{n} /// ... x `rounds` /// /// These three modes control manufacturing. #[structopt(long="mode", default_value = "MasterToN")] pub mode: node_transaction_factory::Mode, /// Number of transactions to generate. In mode `MasterNToNToM` this is /// the number of transactions per round. #[structopt(long="num", default_value = "8")] pub num: u64, #[allow(missing_docs)] #[structopt(flatten)] pub shared_params: SharedParams, #[allow(missing_docs)] #[structopt(flatten)] pub import_params: ImportParams, } /// Parse command line arguments into service configuration. pub fn run(args: I, exit: E, version: sc_cli::VersionInfo) -> error::Result<()> where I: IntoIterator, T: Into + Clone, E: IntoExit, { type Config = Configuration<(), A, B>; match parse_and_prepare::(&version, "substrate-node", args) { ParseAndPrepare::Run(cmd) => cmd.run(load_spec, exit, |exit, _cli_args, _custom_args, mut config: Config<_, _>| { info!("{}", version.name); info!(" version {}", config.full_version()); info!(" by Parity Technologies, 2017-2019"); info!("Chain specification: {}", config.chain_spec.name()); info!("Node name: {}", config.name); info!("Roles: {}", display_role(&config)); let runtime = RuntimeBuilder::new() .thread_name("main-tokio-") .threaded_scheduler() .enable_all() .build() .map_err(|e| format!("{:?}", e))?; config.tasks_executor = { let runtime_handle = runtime.handle().clone(); Some(Box::new(move |fut| { runtime_handle.spawn(fut); })) }; match config.roles { ServiceRoles::LIGHT => run_until_exit( runtime, service::new_light(config)?, exit ), _ => run_until_exit( runtime, service::new_full(config)?, exit ), } }), ParseAndPrepare::BuildSpec(cmd) => cmd.run::(load_spec), ParseAndPrepare::ExportBlocks(cmd) => cmd.run_with_builder(|config: Config<_, _>| Ok(new_full_start!(config).0), load_spec, exit), ParseAndPrepare::ImportBlocks(cmd) => cmd.run_with_builder(|config: Config<_, _>| Ok(new_full_start!(config).0), load_spec, exit), ParseAndPrepare::CheckBlock(cmd) => cmd.run_with_builder(|config: Config<_, _>| Ok(new_full_start!(config).0), load_spec, exit), ParseAndPrepare::PurgeChain(cmd) => cmd.run(load_spec), ParseAndPrepare::RevertChain(cmd) => cmd.run_with_builder(|config: Config<_, _>| Ok(new_full_start!(config).0), load_spec), ParseAndPrepare::CustomCommand(CustomSubcommands::Factory(cli_args)) => { let mut config: Config<_, _> = sc_cli::create_config_with_db_path( load_spec, &cli_args.shared_params, &version, None, )?; sc_cli::fill_import_params( &mut config, &cli_args.import_params, ServiceRoles::FULL, cli_args.shared_params.dev, )?; match ChainSpec::from(config.chain_spec.id()) { Some(ref c) if c == &ChainSpec::Development || c == &ChainSpec::LocalTestnet => {}, _ => panic!("Factory is only supported for development and local testnet."), } let factory_state = FactoryState::new( cli_args.mode.clone(), cli_args.num, cli_args.rounds, ); let service_builder = new_full_start!(config).0; node_transaction_factory::factory::, _, _, _, _, _>( factory_state, service_builder.client(), service_builder.select_chain() .expect("The select_chain is always initialized by new_full_start!; QED") ).map_err(|e| format!("Error in transaction factory: {}", e))?; Ok(()) } } } fn run_until_exit( mut runtime: Runtime, service: T, e: E, ) -> error::Result<()> where T: AbstractService, E: IntoExit, { let (exit_send, exit) = oneshot::channel(); let informant = sc_cli::informant::build(&service); let handle = runtime.spawn(select(exit, informant)); // we eagerly drop the service so that the internal exit future is fired, // but we need to keep holding a reference to the global telemetry guard let _telemetry = service.telemetry(); let exit = e.into_exit(); let service_res = runtime.block_on(select(service, exit)); let _ = exit_send.send(()); runtime.block_on(handle); match service_res { Either::Left((res, _)) => res.map_err(error::Error::Service), Either::Right((_, _)) => Ok(()) } }