mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-07-04 01:07:23 +00:00
8683bbeefb
Closes #2326. This PR both fixes a logic bug and replaces an incorrect name. ## Bug Fix: Respecting custom genesis builder Prior to this PR the standard logic for creating a genesis block was repeated inside of cumulus. This PR removes that duplicated logic, and calls into the proper `BuildGenesisBlock` implementation. One consequence is that if the genesis block has already been initialized, it will not be re-created, but rather read from the database like it is for other node invocations. So you need to watch out for old unpurged data during the development process. Offchain tools may need to be updated accordingly. I've already filed https://github.com/paritytech/zombienet/issues/1519 ## Rename: It doesn't export state. It exports head data. The name export-genesis-state was always wrong, nad it's never too late to right a wrong. I've changed the name of the struct to `ExportGenesisHeadCommand`. There is still the question of what to do with individual nodes' public CLIs. I have updated the parachain template to a reasonable default that preserves compatibility with tools that will expect `export-genesis-state` to still work. And I've chosen not to modify the public CLIs of any other nodes in the repo. I'll leave it up to their individual owners/maintains to decide whether that is appropriate. --------- Co-authored-by: Joshy Orndorff <git-user-email.h0ly5@simplelogin.com> Co-authored-by: Bastian Köcher <git@kchr.de> Co-authored-by: Bastian Köcher <info@kchr.de>
452 lines
12 KiB
Rust
452 lines
12 KiB
Rust
// Copyright (C) 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/>.
|
|
|
|
//! Cumulus CLI library.
|
|
|
|
#![warn(missing_docs)]
|
|
|
|
use std::{
|
|
fs,
|
|
io::{self, Write},
|
|
net::SocketAddr,
|
|
path::PathBuf,
|
|
sync::Arc,
|
|
};
|
|
|
|
use codec::Encode;
|
|
use sc_chain_spec::ChainSpec;
|
|
use sc_client_api::HeaderBackend;
|
|
use sc_service::{
|
|
config::{PrometheusConfig, TelemetryEndpoints},
|
|
BasePath, TransactionPoolOptions,
|
|
};
|
|
use sp_core::hexdisplay::HexDisplay;
|
|
use sp_runtime::traits::{Block as BlockT, Zero};
|
|
use url::Url;
|
|
|
|
/// The `purge-chain` command used to remove the whole chain: the parachain and the relay chain.
|
|
#[derive(Debug, clap::Parser)]
|
|
#[group(skip)]
|
|
pub struct PurgeChainCmd {
|
|
/// The base struct of the purge-chain command.
|
|
#[command(flatten)]
|
|
pub base: sc_cli::PurgeChainCmd,
|
|
|
|
/// Only delete the para chain database
|
|
#[arg(long, aliases = &["para"])]
|
|
pub parachain: bool,
|
|
|
|
/// Only delete the relay chain database
|
|
#[arg(long, aliases = &["relay"])]
|
|
pub relaychain: bool,
|
|
}
|
|
|
|
impl PurgeChainCmd {
|
|
/// Run the purge command
|
|
pub fn run(
|
|
&self,
|
|
para_config: sc_service::Configuration,
|
|
relay_config: sc_service::Configuration,
|
|
) -> sc_cli::Result<()> {
|
|
let databases = match (self.parachain, self.relaychain) {
|
|
(true, true) | (false, false) => {
|
|
vec![("parachain", para_config.database), ("relaychain", relay_config.database)]
|
|
},
|
|
(true, false) => vec![("parachain", para_config.database)],
|
|
(false, true) => vec![("relaychain", relay_config.database)],
|
|
};
|
|
|
|
let db_paths = databases
|
|
.iter()
|
|
.map(|(chain_label, database)| {
|
|
database.path().ok_or_else(|| {
|
|
sc_cli::Error::Input(format!(
|
|
"Cannot purge custom database implementation of: {}",
|
|
chain_label,
|
|
))
|
|
})
|
|
})
|
|
.collect::<sc_cli::Result<Vec<_>>>()?;
|
|
|
|
if !self.base.yes {
|
|
for db_path in &db_paths {
|
|
println!("{}", db_path.display());
|
|
}
|
|
print!("Are you sure to remove? [y/N]: ");
|
|
io::stdout().flush().expect("failed to flush stdout");
|
|
|
|
let mut input = String::new();
|
|
io::stdin().read_line(&mut input)?;
|
|
let input = input.trim();
|
|
|
|
match input.chars().next() {
|
|
Some('y') | Some('Y') => {},
|
|
_ => {
|
|
println!("Aborted");
|
|
return Ok(())
|
|
},
|
|
}
|
|
}
|
|
|
|
for db_path in &db_paths {
|
|
match fs::remove_dir_all(db_path) {
|
|
Ok(_) => {
|
|
println!("{:?} removed.", &db_path);
|
|
},
|
|
Err(ref err) if err.kind() == io::ErrorKind::NotFound => {
|
|
eprintln!("{:?} did not exist.", &db_path);
|
|
},
|
|
Err(err) => return Err(err.into()),
|
|
}
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
impl sc_cli::CliConfiguration for PurgeChainCmd {
|
|
fn shared_params(&self) -> &sc_cli::SharedParams {
|
|
&self.base.shared_params
|
|
}
|
|
|
|
fn database_params(&self) -> Option<&sc_cli::DatabaseParams> {
|
|
Some(&self.base.database_params)
|
|
}
|
|
}
|
|
|
|
/// Command for exporting the genesis head data of the parachain
|
|
#[derive(Debug, clap::Parser)]
|
|
pub struct ExportGenesisHeadCommand {
|
|
/// Output file name or stdout if unspecified.
|
|
#[arg()]
|
|
pub output: Option<PathBuf>,
|
|
|
|
/// Write output in binary. Default is to write in hex.
|
|
#[arg(short, long)]
|
|
pub raw: bool,
|
|
|
|
#[allow(missing_docs)]
|
|
#[command(flatten)]
|
|
pub shared_params: sc_cli::SharedParams,
|
|
}
|
|
|
|
impl ExportGenesisHeadCommand {
|
|
/// Run the export-genesis-head command
|
|
pub fn run<B, C>(&self, client: Arc<C>) -> sc_cli::Result<()>
|
|
where
|
|
B: BlockT,
|
|
C: HeaderBackend<B> + 'static,
|
|
{
|
|
let genesis_hash = client.hash(Zero::zero())?.ok_or(sc_cli::Error::Client(
|
|
sp_blockchain::Error::Backend(
|
|
"Failed to lookup genesis block hash when exporting genesis head data.".into(),
|
|
),
|
|
))?;
|
|
let genesis_header = client.header(genesis_hash)?.ok_or(sc_cli::Error::Client(
|
|
sp_blockchain::Error::Backend(
|
|
"Failed to lookup genesis header by hash when exporting genesis head data.".into(),
|
|
),
|
|
))?;
|
|
|
|
let raw_header = genesis_header.encode();
|
|
let output_buf = if self.raw {
|
|
raw_header
|
|
} else {
|
|
format!("0x{:?}", HexDisplay::from(&genesis_header.encode())).into_bytes()
|
|
};
|
|
|
|
if let Some(output) = &self.output {
|
|
fs::write(output, output_buf)?;
|
|
} else {
|
|
io::stdout().write_all(&output_buf)?;
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
impl sc_cli::CliConfiguration for ExportGenesisHeadCommand {
|
|
fn shared_params(&self) -> &sc_cli::SharedParams {
|
|
&self.shared_params
|
|
}
|
|
|
|
fn base_path(&self) -> sc_cli::Result<Option<BasePath>> {
|
|
// As we are just exporting the genesis wasm a tmp database is enough.
|
|
//
|
|
// As otherwise we may "pollute" the global base path.
|
|
Ok(Some(BasePath::new_temp_dir()?))
|
|
}
|
|
}
|
|
|
|
/// Command for exporting the genesis wasm file.
|
|
#[derive(Debug, clap::Parser)]
|
|
pub struct ExportGenesisWasmCommand {
|
|
/// Output file name or stdout if unspecified.
|
|
#[arg()]
|
|
pub output: Option<PathBuf>,
|
|
|
|
/// Write output in binary. Default is to write in hex.
|
|
#[arg(short, long)]
|
|
pub raw: bool,
|
|
|
|
#[allow(missing_docs)]
|
|
#[command(flatten)]
|
|
pub shared_params: sc_cli::SharedParams,
|
|
}
|
|
|
|
impl ExportGenesisWasmCommand {
|
|
/// Run the export-genesis-wasm command
|
|
pub fn run(&self, chain_spec: &dyn ChainSpec) -> sc_cli::Result<()> {
|
|
let raw_wasm_blob = extract_genesis_wasm(chain_spec)?;
|
|
let output_buf = if self.raw {
|
|
raw_wasm_blob
|
|
} else {
|
|
format!("0x{:?}", HexDisplay::from(&raw_wasm_blob)).into_bytes()
|
|
};
|
|
|
|
if let Some(output) = &self.output {
|
|
fs::write(output, output_buf)?;
|
|
} else {
|
|
io::stdout().write_all(&output_buf)?;
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
/// Extract the genesis code from a given ChainSpec.
|
|
pub fn extract_genesis_wasm(chain_spec: &dyn ChainSpec) -> sc_cli::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())
|
|
}
|
|
|
|
impl sc_cli::CliConfiguration for ExportGenesisWasmCommand {
|
|
fn shared_params(&self) -> &sc_cli::SharedParams {
|
|
&self.shared_params
|
|
}
|
|
|
|
fn base_path(&self) -> sc_cli::Result<Option<BasePath>> {
|
|
// As we are just exporting the genesis wasm a tmp database is enough.
|
|
//
|
|
// As otherwise we may "pollute" the global base path.
|
|
Ok(Some(BasePath::new_temp_dir()?))
|
|
}
|
|
}
|
|
|
|
fn validate_relay_chain_url(arg: &str) -> Result<Url, String> {
|
|
let url = Url::parse(arg).map_err(|e| e.to_string())?;
|
|
|
|
let scheme = url.scheme();
|
|
if scheme == "ws" || scheme == "wss" {
|
|
Ok(url)
|
|
} else {
|
|
Err(format!(
|
|
"'{}' URL scheme not supported. Only websocket RPC is currently supported",
|
|
url.scheme()
|
|
))
|
|
}
|
|
}
|
|
|
|
/// The `run` command used to run a node.
|
|
#[derive(Debug, clap::Parser)]
|
|
#[group(skip)]
|
|
pub struct RunCmd {
|
|
/// The cumulus RunCmd inherents from sc_cli's
|
|
#[command(flatten)]
|
|
pub base: sc_cli::RunCmd,
|
|
|
|
/// Run node as collator.
|
|
///
|
|
/// Note that this is the same as running with `--validator`.
|
|
#[arg(long, conflicts_with = "validator")]
|
|
pub collator: bool,
|
|
|
|
/// Creates a less resource-hungry node that retrieves relay chain data from an RPC endpoint.
|
|
///
|
|
/// The provided URLs should point to RPC endpoints of the relay chain.
|
|
/// This node connects to the remote nodes following the order they were specified in. If the
|
|
/// connection fails, it attempts to connect to the next endpoint in the list.
|
|
///
|
|
/// Note: This option doesn't stop the node from connecting to the relay chain network but
|
|
/// reduces bandwidth use.
|
|
#[arg(
|
|
long,
|
|
value_parser = validate_relay_chain_url,
|
|
num_args = 0..,
|
|
alias = "relay-chain-rpc-url"
|
|
)]
|
|
pub relay_chain_rpc_urls: Vec<Url>,
|
|
|
|
/// EXPERIMENTAL: Embed a light client for the relay chain. Only supported for full-nodes.
|
|
/// Will use the specified relay chain chainspec.
|
|
#[arg(long, conflicts_with_all = ["relay_chain_rpc_urls", "collator"])]
|
|
pub relay_chain_light_client: bool,
|
|
}
|
|
|
|
impl RunCmd {
|
|
/// Create a [`NormalizedRunCmd`] which merges the `collator` cli argument into `validator` to
|
|
/// have only one.
|
|
pub fn normalize(&self) -> NormalizedRunCmd {
|
|
let mut new_base = self.base.clone();
|
|
|
|
new_base.validator = self.base.validator || self.collator;
|
|
|
|
NormalizedRunCmd { base: new_base }
|
|
}
|
|
|
|
/// Create [`CollatorOptions`] representing options only relevant to parachain collator nodes
|
|
pub fn collator_options(&self) -> CollatorOptions {
|
|
let relay_chain_mode =
|
|
match (self.relay_chain_light_client, !self.relay_chain_rpc_urls.is_empty()) {
|
|
(true, _) => RelayChainMode::LightClient,
|
|
(_, true) => RelayChainMode::ExternalRpc(self.relay_chain_rpc_urls.clone()),
|
|
_ => RelayChainMode::Embedded,
|
|
};
|
|
|
|
CollatorOptions { relay_chain_mode }
|
|
}
|
|
}
|
|
|
|
/// Possible modes for the relay chain to operate in.
|
|
#[derive(Clone, Debug)]
|
|
pub enum RelayChainMode {
|
|
/// Spawn embedded relay chain node
|
|
Embedded,
|
|
/// Connect to remote relay chain node via websocket RPC
|
|
ExternalRpc(Vec<Url>),
|
|
/// Spawn embedded relay chain light client
|
|
LightClient,
|
|
}
|
|
|
|
/// Options only relevant for collator nodes
|
|
#[derive(Clone, Debug)]
|
|
pub struct CollatorOptions {
|
|
/// How this collator retrieves relay chain information
|
|
pub relay_chain_mode: RelayChainMode,
|
|
}
|
|
|
|
/// A non-redundant version of the `RunCmd` that sets the `validator` field when the
|
|
/// original `RunCmd` had the `collator` field.
|
|
/// This is how we make `--collator` imply `--validator`.
|
|
pub struct NormalizedRunCmd {
|
|
/// The cumulus RunCmd inherents from sc_cli's
|
|
pub base: sc_cli::RunCmd,
|
|
}
|
|
|
|
impl sc_cli::CliConfiguration for NormalizedRunCmd {
|
|
fn shared_params(&self) -> &sc_cli::SharedParams {
|
|
self.base.shared_params()
|
|
}
|
|
|
|
fn import_params(&self) -> Option<&sc_cli::ImportParams> {
|
|
self.base.import_params()
|
|
}
|
|
|
|
fn network_params(&self) -> Option<&sc_cli::NetworkParams> {
|
|
self.base.network_params()
|
|
}
|
|
|
|
fn keystore_params(&self) -> Option<&sc_cli::KeystoreParams> {
|
|
self.base.keystore_params()
|
|
}
|
|
|
|
fn offchain_worker_params(&self) -> Option<&sc_cli::OffchainWorkerParams> {
|
|
self.base.offchain_worker_params()
|
|
}
|
|
|
|
fn node_name(&self) -> sc_cli::Result<String> {
|
|
self.base.node_name()
|
|
}
|
|
|
|
fn dev_key_seed(&self, is_dev: bool) -> sc_cli::Result<Option<String>> {
|
|
self.base.dev_key_seed(is_dev)
|
|
}
|
|
|
|
fn telemetry_endpoints(
|
|
&self,
|
|
chain_spec: &Box<dyn sc_cli::ChainSpec>,
|
|
) -> sc_cli::Result<Option<TelemetryEndpoints>> {
|
|
self.base.telemetry_endpoints(chain_spec)
|
|
}
|
|
|
|
fn role(&self, is_dev: bool) -> sc_cli::Result<sc_cli::Role> {
|
|
self.base.role(is_dev)
|
|
}
|
|
|
|
fn force_authoring(&self) -> sc_cli::Result<bool> {
|
|
self.base.force_authoring()
|
|
}
|
|
|
|
fn prometheus_config(
|
|
&self,
|
|
default_listen_port: u16,
|
|
chain_spec: &Box<dyn sc_cli::ChainSpec>,
|
|
) -> sc_cli::Result<Option<PrometheusConfig>> {
|
|
self.base.prometheus_config(default_listen_port, chain_spec)
|
|
}
|
|
|
|
fn disable_grandpa(&self) -> sc_cli::Result<bool> {
|
|
self.base.disable_grandpa()
|
|
}
|
|
|
|
fn rpc_max_connections(&self) -> sc_cli::Result<u32> {
|
|
self.base.rpc_max_connections()
|
|
}
|
|
|
|
fn rpc_cors(&self, is_dev: bool) -> sc_cli::Result<Option<Vec<String>>> {
|
|
self.base.rpc_cors(is_dev)
|
|
}
|
|
|
|
fn rpc_addr(&self, default_listen_port: u16) -> sc_cli::Result<Option<SocketAddr>> {
|
|
self.base.rpc_addr(default_listen_port)
|
|
}
|
|
|
|
fn rpc_methods(&self) -> sc_cli::Result<sc_service::config::RpcMethods> {
|
|
self.base.rpc_methods()
|
|
}
|
|
|
|
fn rpc_max_request_size(&self) -> sc_cli::Result<u32> {
|
|
Ok(self.base.rpc_max_request_size)
|
|
}
|
|
|
|
fn rpc_max_response_size(&self) -> sc_cli::Result<u32> {
|
|
Ok(self.base.rpc_max_response_size)
|
|
}
|
|
|
|
fn rpc_max_subscriptions_per_connection(&self) -> sc_cli::Result<u32> {
|
|
Ok(self.base.rpc_max_subscriptions_per_connection)
|
|
}
|
|
|
|
fn transaction_pool(&self, is_dev: bool) -> sc_cli::Result<TransactionPoolOptions> {
|
|
self.base.transaction_pool(is_dev)
|
|
}
|
|
|
|
fn max_runtime_instances(&self) -> sc_cli::Result<Option<usize>> {
|
|
self.base.max_runtime_instances()
|
|
}
|
|
|
|
fn runtime_cache_size(&self) -> sc_cli::Result<u8> {
|
|
self.base.runtime_cache_size()
|
|
}
|
|
|
|
fn base_path(&self) -> sc_cli::Result<Option<BasePath>> {
|
|
self.base.base_path()
|
|
}
|
|
}
|