mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-04-29 03:17:56 +00:00
3b51e53206
* Initial commit
Forked at: 5bbab68f02
Parent branch: origin/master
* Add --collator flag as alternative to --validator
* Apply suggestions from code review
Co-authored-by: Bastian Köcher <bkchr@users.noreply.github.com>
Co-authored-by: Bastian Köcher <bkchr@users.noreply.github.com>
434 lines
12 KiB
Rust
434 lines
12 KiB
Rust
// Copyright 2019 Parity Technologies (UK) Ltd.
|
|
// This file is part of Cumulus.
|
|
|
|
// Cumulus 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.
|
|
|
|
// Cumulus 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 <http://www.gnu.org/licenses/>.
|
|
|
|
use crate::{
|
|
chain_spec,
|
|
cli::{Cli, RelayChainCli, Subcommand},
|
|
};
|
|
use codec::Encode;
|
|
use cumulus_primitives::ParaId;
|
|
use log::info;
|
|
use parachain_runtime::Block;
|
|
use polkadot_parachain::primitives::AccountIdConversion;
|
|
use sc_cli::{
|
|
ChainSpec, CliConfiguration, DefaultConfigurationValues, ImportParams, KeystoreParams,
|
|
NetworkParams, Result, RuntimeVersion, SharedParams, SubstrateCli,
|
|
};
|
|
use sc_service::config::{BasePath, PrometheusConfig};
|
|
use sp_core::hexdisplay::HexDisplay;
|
|
use sp_runtime::traits::{Block as BlockT, Hash as HashT, Header as HeaderT, Zero};
|
|
use std::{io::Write, net::SocketAddr, sync::Arc};
|
|
|
|
fn load_spec(
|
|
id: &str,
|
|
para_id: ParaId,
|
|
) -> std::result::Result<Box<dyn sc_service::ChainSpec>, String> {
|
|
match id {
|
|
"staging" => Ok(Box::new(chain_spec::staging_test_net(para_id))),
|
|
"tick" => Ok(Box::new(chain_spec::ChainSpec::from_json_bytes(
|
|
&include_bytes!("../res/tick.json")[..],
|
|
)?)),
|
|
"trick" => Ok(Box::new(chain_spec::ChainSpec::from_json_bytes(
|
|
&include_bytes!("../res/trick.json")[..],
|
|
)?)),
|
|
"track" => Ok(Box::new(chain_spec::ChainSpec::from_json_bytes(
|
|
&include_bytes!("../res/track.json")[..],
|
|
)?)),
|
|
"contracts" => Ok(Box::new(chain_spec::get_contracts_chain_spec(para_id))),
|
|
"" => Ok(Box::new(chain_spec::get_chain_spec(para_id))),
|
|
path => Ok(Box::new(chain_spec::ChainSpec::from_json_file(
|
|
path.into(),
|
|
)?)),
|
|
}
|
|
}
|
|
|
|
impl SubstrateCli for Cli {
|
|
fn impl_name() -> String {
|
|
"Cumulus Test Parachain Collator".into()
|
|
}
|
|
|
|
fn impl_version() -> String {
|
|
env!("SUBSTRATE_CLI_IMPL_VERSION").into()
|
|
}
|
|
|
|
fn description() -> String {
|
|
format!(
|
|
"Cumulus test parachain collator\n\nThe command-line arguments provided first will be \
|
|
passed to the parachain node, while the arguments provided after -- will be passed \
|
|
to the relaychain node.\n\n\
|
|
{} [parachain-args] -- [relaychain-args]",
|
|
Self::executable_name()
|
|
)
|
|
}
|
|
|
|
fn author() -> String {
|
|
env!("CARGO_PKG_AUTHORS").into()
|
|
}
|
|
|
|
fn support_url() -> String {
|
|
"https://github.com/paritytech/cumulus/issues/new".into()
|
|
}
|
|
|
|
fn copyright_start_year() -> i32 {
|
|
2017
|
|
}
|
|
|
|
fn load_spec(&self, id: &str) -> std::result::Result<Box<dyn sc_service::ChainSpec>, String> {
|
|
load_spec(id, self.run.parachain_id.unwrap_or(100).into())
|
|
}
|
|
|
|
fn native_runtime_version(_: &Box<dyn ChainSpec>) -> &'static RuntimeVersion {
|
|
¶chain_runtime::VERSION
|
|
}
|
|
}
|
|
|
|
impl SubstrateCli for RelayChainCli {
|
|
fn impl_name() -> String {
|
|
"Cumulus Test Parachain Collator".into()
|
|
}
|
|
|
|
fn impl_version() -> String {
|
|
env!("SUBSTRATE_CLI_IMPL_VERSION").into()
|
|
}
|
|
|
|
fn description() -> String {
|
|
"Cumulus test parachain collator\n\nThe command-line arguments provided first will be \
|
|
passed to the parachain node, while the arguments provided after -- will be passed \
|
|
to the relaychain node.\n\n\
|
|
rococo-collator [parachain-args] -- [relaychain-args]"
|
|
.into()
|
|
}
|
|
|
|
fn author() -> String {
|
|
env!("CARGO_PKG_AUTHORS").into()
|
|
}
|
|
|
|
fn support_url() -> String {
|
|
"https://github.com/paritytech/cumulus/issues/new".into()
|
|
}
|
|
|
|
fn copyright_start_year() -> i32 {
|
|
2017
|
|
}
|
|
|
|
fn load_spec(&self, id: &str) -> std::result::Result<Box<dyn sc_service::ChainSpec>, String> {
|
|
polkadot_cli::Cli::from_iter([RelayChainCli::executable_name().to_string()].iter())
|
|
.load_spec(id)
|
|
}
|
|
|
|
fn native_runtime_version(chain_spec: &Box<dyn ChainSpec>) -> &'static RuntimeVersion {
|
|
polkadot_cli::Cli::native_runtime_version(chain_spec)
|
|
}
|
|
}
|
|
|
|
pub fn generate_genesis_state(chain_spec: &Box<dyn sc_service::ChainSpec>) -> Result<Block> {
|
|
let storage = chain_spec.build_storage()?;
|
|
|
|
let child_roots = storage.children_default.iter().map(|(sk, child_content)| {
|
|
let state_root = <<<Block as BlockT>::Header as HeaderT>::Hashing as HashT>::trie_root(
|
|
child_content.data.clone().into_iter().collect(),
|
|
);
|
|
(sk.clone(), state_root.encode())
|
|
});
|
|
let state_root = <<<Block as BlockT>::Header as HeaderT>::Hashing as HashT>::trie_root(
|
|
storage.top.clone().into_iter().chain(child_roots).collect(),
|
|
);
|
|
|
|
let extrinsics_root =
|
|
<<<Block as BlockT>::Header as HeaderT>::Hashing as HashT>::trie_root(Vec::new());
|
|
|
|
Ok(Block::new(
|
|
<<Block as BlockT>::Header as HeaderT>::new(
|
|
Zero::zero(),
|
|
extrinsics_root,
|
|
state_root,
|
|
Default::default(),
|
|
Default::default(),
|
|
),
|
|
Default::default(),
|
|
))
|
|
}
|
|
|
|
fn extract_genesis_wasm(chain_spec: &Box<dyn sc_service::ChainSpec>) -> Result<Vec<u8>> {
|
|
let mut storage = chain_spec.build_storage()?;
|
|
|
|
storage
|
|
.top
|
|
.remove(sp_core::storage::well_known_keys::CODE)
|
|
.ok_or_else(|| "Could not find wasm file in genesis state!".into())
|
|
}
|
|
|
|
fn use_contracts_runtime(chain_spec: &Box<dyn ChainSpec>) -> bool {
|
|
chain_spec.id().starts_with("trick") || chain_spec.id().starts_with("contracts")
|
|
}
|
|
|
|
/// Parse command line arguments into service configuration.
|
|
pub fn run() -> Result<()> {
|
|
let cli = Cli::from_args();
|
|
|
|
match &cli.subcommand {
|
|
Some(Subcommand::Base(subcommand)) => {
|
|
let runner = cli.create_runner(subcommand)?;
|
|
|
|
if use_contracts_runtime(&runner.config().chain_spec) {
|
|
runner.run_subcommand(subcommand, |mut config| {
|
|
let params = crate::service::new_partial::<
|
|
parachain_contracts_runtime::RuntimeApi,
|
|
crate::service::ContractsRuntimeExecutor,
|
|
>(&mut config)?;
|
|
|
|
Ok((
|
|
params.client,
|
|
params.backend,
|
|
params.import_queue,
|
|
params.task_manager,
|
|
))
|
|
})
|
|
} else {
|
|
runner.run_subcommand(subcommand, |mut config| {
|
|
let params = crate::service::new_partial::<
|
|
parachain_runtime::RuntimeApi,
|
|
crate::service::RuntimeExecutor,
|
|
>(&mut config)?;
|
|
|
|
Ok((
|
|
params.client,
|
|
params.backend,
|
|
params.import_queue,
|
|
params.task_manager,
|
|
))
|
|
})
|
|
}
|
|
}
|
|
Some(Subcommand::ExportGenesisState(params)) => {
|
|
sc_cli::init_logger("");
|
|
|
|
let block = generate_genesis_state(&load_spec(
|
|
¶ms.chain.clone().unwrap_or_default(),
|
|
params.parachain_id.into(),
|
|
)?)?;
|
|
let header_hex = format!("0x{:?}", HexDisplay::from(&block.header().encode()));
|
|
|
|
if let Some(output) = ¶ms.output {
|
|
std::fs::write(output, header_hex)?;
|
|
} else {
|
|
print!("{}", header_hex);
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
Some(Subcommand::ExportGenesisWasm(params)) => {
|
|
sc_cli::init_logger("");
|
|
|
|
let wasm_file =
|
|
extract_genesis_wasm(&cli.load_spec(¶ms.chain.clone().unwrap_or_default())?)?;
|
|
|
|
if let Some(output) = ¶ms.output {
|
|
std::fs::write(output, wasm_file)?;
|
|
} else {
|
|
std::io::stdout().write_all(&wasm_file)?;
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
None => {
|
|
let runner = cli.create_runner(&*cli.run)?;
|
|
|
|
runner.run_node_until_exit(|config| {
|
|
// TODO
|
|
let key = Arc::new(sp_core::Pair::generate().0);
|
|
|
|
let extension = chain_spec::Extensions::try_get(&config.chain_spec);
|
|
let relay_chain_id = extension.map(|e| e.relay_chain.clone());
|
|
let para_id = extension.map(|e| e.para_id);
|
|
|
|
let polkadot_cli = RelayChainCli::new(
|
|
config.base_path.as_ref().map(|x| x.path().join("polkadot")),
|
|
relay_chain_id,
|
|
[RelayChainCli::executable_name().to_string()]
|
|
.iter()
|
|
.chain(cli.relaychain_args.iter()),
|
|
);
|
|
|
|
let id = ParaId::from(cli.run.parachain_id.or(para_id).unwrap_or(100));
|
|
|
|
let parachain_account =
|
|
AccountIdConversion::<polkadot_primitives::v0::AccountId>::into_account(&id);
|
|
|
|
let block =
|
|
generate_genesis_state(&config.chain_spec).map_err(|e| format!("{:?}", e))?;
|
|
let genesis_state = format!("0x{:?}", HexDisplay::from(&block.header().encode()));
|
|
|
|
let task_executor = config.task_executor.clone();
|
|
let polkadot_config =
|
|
SubstrateCli::create_configuration(&polkadot_cli, &polkadot_cli, task_executor)
|
|
.map_err(|err| format!("Relay chain argument error: {}", err))?;
|
|
let collator = cli.run.base.validator || cli.collator;
|
|
|
|
info!("Parachain id: {:?}", id);
|
|
info!("Parachain Account: {}", parachain_account);
|
|
info!("Parachain genesis state: {}", genesis_state);
|
|
info!(
|
|
"Is collating: {}",
|
|
if collator { "yes" } else { "no" }
|
|
);
|
|
|
|
if use_contracts_runtime(&config.chain_spec) {
|
|
crate::service::start_contracts_node(
|
|
config,
|
|
key,
|
|
polkadot_config,
|
|
id,
|
|
collator,
|
|
false,
|
|
)
|
|
} else {
|
|
crate::service::start_node(
|
|
config,
|
|
key,
|
|
polkadot_config,
|
|
id,
|
|
collator,
|
|
false,
|
|
)
|
|
.map(|r| r.0)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
}
|
|
|
|
impl DefaultConfigurationValues for RelayChainCli {
|
|
fn p2p_listen_port() -> u16 {
|
|
30334
|
|
}
|
|
|
|
fn rpc_ws_listen_port() -> u16 {
|
|
9945
|
|
}
|
|
|
|
fn rpc_http_listen_port() -> u16 {
|
|
9934
|
|
}
|
|
|
|
fn prometheus_listen_port() -> u16 {
|
|
9616
|
|
}
|
|
}
|
|
|
|
impl CliConfiguration<Self> for RelayChainCli {
|
|
fn shared_params(&self) -> &SharedParams {
|
|
self.base.base.shared_params()
|
|
}
|
|
|
|
fn import_params(&self) -> Option<&ImportParams> {
|
|
self.base.base.import_params()
|
|
}
|
|
|
|
fn network_params(&self) -> Option<&NetworkParams> {
|
|
self.base.base.network_params()
|
|
}
|
|
|
|
fn keystore_params(&self) -> Option<&KeystoreParams> {
|
|
self.base.base.keystore_params()
|
|
}
|
|
|
|
fn base_path(&self) -> Result<Option<BasePath>> {
|
|
Ok(self
|
|
.shared_params()
|
|
.base_path()
|
|
.or_else(|| self.base_path.clone().map(Into::into)))
|
|
}
|
|
|
|
fn rpc_http(&self, default_listen_port: u16) -> Result<Option<SocketAddr>> {
|
|
self.base.base.rpc_http(default_listen_port)
|
|
}
|
|
|
|
fn rpc_ipc(&self) -> Result<Option<String>> {
|
|
self.base.base.rpc_ipc()
|
|
}
|
|
|
|
fn rpc_ws(&self, default_listen_port: u16) -> Result<Option<SocketAddr>> {
|
|
self.base.base.rpc_ws(default_listen_port)
|
|
}
|
|
|
|
fn prometheus_config(&self, default_listen_port: u16) -> Result<Option<PrometheusConfig>> {
|
|
self.base.base.prometheus_config(default_listen_port)
|
|
}
|
|
|
|
fn init<C: SubstrateCli>(&self) -> Result<()> {
|
|
unreachable!("PolkadotCli is never initialized; qed");
|
|
}
|
|
|
|
fn chain_id(&self, is_dev: bool) -> Result<String> {
|
|
let chain_id = self.base.base.chain_id(is_dev)?;
|
|
|
|
Ok(if chain_id.is_empty() {
|
|
self.chain_id.clone().unwrap_or_default()
|
|
} else {
|
|
chain_id
|
|
})
|
|
}
|
|
|
|
fn role(&self, is_dev: bool) -> Result<sc_service::Role> {
|
|
self.base.base.role(is_dev)
|
|
}
|
|
|
|
fn transaction_pool(&self) -> Result<sc_service::config::TransactionPoolOptions> {
|
|
self.base.base.transaction_pool()
|
|
}
|
|
|
|
fn state_cache_child_ratio(&self) -> Result<Option<usize>> {
|
|
self.base.base.state_cache_child_ratio()
|
|
}
|
|
|
|
fn rpc_methods(&self) -> Result<sc_service::config::RpcMethods> {
|
|
self.base.base.rpc_methods()
|
|
}
|
|
|
|
fn rpc_ws_max_connections(&self) -> Result<Option<usize>> {
|
|
self.base.base.rpc_ws_max_connections()
|
|
}
|
|
|
|
fn rpc_cors(&self, is_dev: bool) -> Result<Option<Vec<String>>> {
|
|
self.base.base.rpc_cors(is_dev)
|
|
}
|
|
|
|
fn telemetry_external_transport(&self) -> Result<Option<sc_service::config::ExtTransport>> {
|
|
self.base.base.telemetry_external_transport()
|
|
}
|
|
|
|
fn default_heap_pages(&self) -> Result<Option<u64>> {
|
|
self.base.base.default_heap_pages()
|
|
}
|
|
|
|
fn force_authoring(&self) -> Result<bool> {
|
|
self.base.base.force_authoring()
|
|
}
|
|
|
|
fn disable_grandpa(&self) -> Result<bool> {
|
|
self.base.base.disable_grandpa()
|
|
}
|
|
|
|
fn max_runtime_instances(&self) -> Result<Option<usize>> {
|
|
self.base.base.max_runtime_instances()
|
|
}
|
|
|
|
fn announce_block(&self) -> Result<bool> {
|
|
self.base.base.announce_block()
|
|
}
|
|
}
|