1const DEFAULT_DEV_BLOCK_TIME_MS: u64 = 3000;
21
22use crate::{
23 chain_spec::DiskChainSpecLoader,
24 common::{
25 chain_spec::{Extensions, LoadSpec},
26 NodeExtraArgs,
27 },
28};
29use chain_spec_builder::ChainSpecBuilder;
30use clap::{Command, CommandFactory, FromArgMatches, ValueEnum};
31use sc_chain_spec::ChainSpec;
32use sc_cli::{
33 CliConfiguration, DefaultConfigurationValues, ImportParams, KeystoreParams, NetworkParams,
34 RpcEndpoint, SharedParams, SubstrateCli,
35};
36use sc_service::{config::PrometheusConfig, BasePath};
37use sc_storage_monitor::StorageMonitorParams;
38use std::{
39 fmt::{Display, Formatter},
40 marker::PhantomData,
41 path::PathBuf,
42};
43pub trait CliConfig {
49 fn impl_version() -> String;
51
52 fn description(executable_name: String) -> String {
54 format!(
55 "The command-line arguments provided first will be passed to the parachain node, \n\
56 and the arguments provided after -- will be passed to the relay chain node. \n\
57 \n\
58 Example: \n\
59 \n\
60 {} [parachain-args] -- [relay-chain-args]",
61 executable_name
62 )
63 }
64
65 fn author() -> String;
67
68 fn support_url() -> String;
70
71 fn copyright_start_year() -> u16;
73}
74
75#[derive(Debug, clap::Subcommand)]
77pub enum Subcommand {
78 #[command(subcommand)]
80 Key(sc_cli::KeySubcommand),
81
82 #[deprecated(
94 note = "build-spec will be removed after 1/06/2025. Use chain-spec-builder instead"
95 )]
96 BuildSpec(sc_cli::BuildSpecCmd),
97
98 CheckBlock(sc_cli::CheckBlockCmd),
100
101 ExportBlocks(sc_cli::ExportBlocksCmd),
103
104 ExportState(sc_cli::ExportStateCmd),
106
107 ImportBlocks(sc_cli::ImportBlocksCmd),
109
110 Revert(sc_cli::RevertCmd),
112
113 ChainSpecBuilder(ChainSpecBuilder),
125
126 PurgeChain(cumulus_client_cli::PurgeChainCmd),
128 #[command(alias = "export-genesis-state")]
130 ExportGenesisHead(cumulus_client_cli::ExportGenesisHeadCommand),
131
132 ExportGenesisWasm(cumulus_client_cli::ExportGenesisWasmCommand),
134
135 #[command(subcommand)]
138 Benchmark(frame_benchmarking_cli::BenchmarkCmd),
139}
140
141#[derive(clap::Parser)]
143#[command(
144 propagate_version = true,
145 args_conflicts_with_subcommands = true,
146 subcommand_negates_reqs = true
147)]
148pub struct Cli<Config: CliConfig> {
149 #[arg(skip)]
150 pub(crate) chain_spec_loader: Option<Box<dyn LoadSpec>>,
151
152 #[command(subcommand)]
154 pub subcommand: Option<Subcommand>,
155
156 #[command(flatten)]
158 pub run: cumulus_client_cli::RunCmd,
159
160 #[command(flatten)]
162 pub storage_monitor: StorageMonitorParams,
163
164 #[arg(long, conflicts_with = "instant_seal")]
175 pub dev_block_time: Option<u64>,
176
177 #[arg(long, conflicts_with = "dev_block_time")]
185 pub instant_seal: bool,
186
187 #[arg(long, conflicts_with = "authoring")]
192 pub experimental_use_slot_based: bool,
193
194 #[arg(long, default_value_t = AuthoringPolicy::Lookahead)]
196 pub authoring: AuthoringPolicy,
197
198 #[arg(long)]
206 pub no_hardware_benchmarks: bool,
207
208 #[arg(long)]
213 pub export_pov_to_path: Option<PathBuf>,
214
215 #[arg(raw = true)]
217 pub relay_chain_args: Vec<String>,
218
219 #[arg(long)]
224 pub enable_statement_store: bool,
225
226 #[arg(long, default_value_t = sc_statement_store::DEFAULT_NETWORK_WORKERS)]
230 pub statement_network_workers: usize,
231
232 #[arg(long, default_value_t = sc_statement_store::DEFAULT_RATE_LIMIT)]
239 pub statement_rate_limit: u32,
240
241 #[arg(long, default_value_t = sc_statement_store::DEFAULT_MAX_TOTAL_STATEMENTS)]
247 pub statement_store_max_total_statements: usize,
248
249 #[arg(long, default_value_t = sc_statement_store::DEFAULT_MAX_TOTAL_SIZE)]
255 pub statement_store_max_total_size: usize,
256
257 #[arg(long, default_value_t = sc_statement_store::DEFAULT_PURGE_AFTER_SEC)]
263 pub statement_store_purge_after_sec: u64,
264
265 #[arg(skip)]
266 pub(crate) _phantom: PhantomData<Config>,
267}
268
269#[derive(Debug, Clone, Copy)]
271pub(crate) enum DevSealMode {
272 InstantSeal,
274 ManualSeal(u64),
277}
278
279#[derive(PartialEq, Debug, ValueEnum, Clone, Copy)]
281pub enum AuthoringPolicy {
282 Lookahead,
285 SlotBased,
289}
290
291impl Display for AuthoringPolicy {
292 fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
293 match self {
294 AuthoringPolicy::Lookahead => write!(f, "lookahead"),
295 AuthoringPolicy::SlotBased => write!(f, "slot-based"),
296 }
297 }
298}
299
300impl<Config: CliConfig> Cli<Config> {
301 pub(crate) fn node_extra_args(&self) -> NodeExtraArgs {
302 NodeExtraArgs {
303 authoring_policy: self
304 .experimental_use_slot_based
305 .then(|| AuthoringPolicy::SlotBased)
306 .unwrap_or(self.authoring),
307 export_pov: self.export_pov_to_path.clone(),
308 max_pov_percentage: self.run.experimental_max_pov_percentage,
309 statement_store_config: self.enable_statement_store.then_some(
310 sc_statement_store::Config {
311 max_total_statements: self.statement_store_max_total_statements,
312 max_total_size: self.statement_store_max_total_size,
313 purge_after_sec: self.statement_store_purge_after_sec,
314 network_workers: self.statement_network_workers,
315 rate_limit: self.statement_rate_limit,
316 },
317 ),
318 storage_monitor: self.storage_monitor.clone(),
319 }
320 }
321
322 pub(crate) fn dev_mode(&self) -> Option<DevSealMode> {
324 if self.instant_seal {
325 Some(DevSealMode::InstantSeal)
326 } else if let Some(dev_block_time) = self.dev_block_time {
327 Some(DevSealMode::ManualSeal(dev_block_time))
328 } else if self.run.base.is_dev().unwrap_or(false) {
329 Some(DevSealMode::ManualSeal(DEFAULT_DEV_BLOCK_TIME_MS))
330 } else {
331 None
332 }
333 }
334}
335
336impl<Config: CliConfig> SubstrateCli for Cli<Config> {
337 fn impl_name() -> String {
338 Self::executable_name()
339 }
340
341 fn impl_version() -> String {
342 Config::impl_version()
343 }
344
345 fn description() -> String {
346 Config::description(Self::executable_name())
347 }
348
349 fn author() -> String {
350 Config::author()
351 }
352
353 fn support_url() -> String {
354 Config::support_url()
355 }
356
357 fn copyright_start_year() -> i32 {
358 Config::copyright_start_year() as i32
359 }
360
361 fn load_spec(&self, id: &str) -> Result<Box<dyn ChainSpec>, String> {
362 match &self.chain_spec_loader {
363 Some(chain_spec_loader) => chain_spec_loader.load_spec(id),
364 None => DiskChainSpecLoader.load_spec(id),
365 }
366 }
367}
368
369#[derive(Debug)]
371pub struct RelayChainCli<Config: CliConfig> {
372 pub base: polkadot_cli::RunCmd,
374
375 pub chain_id: Option<String>,
377
378 pub base_path: Option<PathBuf>,
380
381 _phantom: PhantomData<Config>,
382}
383
384impl<Config: CliConfig> RelayChainCli<Config> {
385 fn polkadot_cmd() -> Command {
386 let help_template = color_print::cformat!(
387 "The arguments that are passed to the relay chain node. \n\
388 \n\
389 <bold><underline>RELAY_CHAIN_ARGS:</></> \n\
390 {{options}}",
391 );
392
393 polkadot_cli::RunCmd::command()
394 .no_binary_name(true)
395 .help_template(help_template)
396 }
397
398 pub fn new<'a>(
400 para_config: &sc_service::Configuration,
401 relay_chain_args: impl Iterator<Item = &'a String>,
402 ) -> Self {
403 let polkadot_cmd = Self::polkadot_cmd();
404 let matches = polkadot_cmd.get_matches_from(relay_chain_args);
405 let base = FromArgMatches::from_arg_matches(&matches).unwrap_or_else(|e| e.exit());
406
407 let extension = Extensions::try_get(&*para_config.chain_spec);
408 let chain_id = extension.map(|e| e.relay_chain());
409
410 let base_path = para_config.base_path.path().join("polkadot");
411 Self { base, chain_id, base_path: Some(base_path), _phantom: Default::default() }
412 }
413}
414
415impl<Config: CliConfig> SubstrateCli for RelayChainCli<Config> {
416 fn impl_name() -> String {
417 Cli::<Config>::impl_name()
418 }
419
420 fn impl_version() -> String {
421 Cli::<Config>::impl_version()
422 }
423
424 fn description() -> String {
425 Cli::<Config>::description()
426 }
427
428 fn author() -> String {
429 Cli::<Config>::author()
430 }
431
432 fn support_url() -> String {
433 Cli::<Config>::support_url()
434 }
435
436 fn copyright_start_year() -> i32 {
437 Cli::<Config>::copyright_start_year()
438 }
439
440 fn load_spec(&self, id: &str) -> std::result::Result<Box<dyn ChainSpec>, String> {
441 polkadot_cli::Cli::from_iter([Self::executable_name()].iter()).load_spec(id)
442 }
443}
444
445impl<Config: CliConfig> DefaultConfigurationValues for RelayChainCli<Config> {
446 fn p2p_listen_port() -> u16 {
447 30334
448 }
449
450 fn rpc_listen_port() -> u16 {
451 9945
452 }
453
454 fn prometheus_listen_port() -> u16 {
455 9616
456 }
457}
458
459impl<Config: CliConfig> CliConfiguration<Self> for RelayChainCli<Config> {
460 fn shared_params(&self) -> &SharedParams {
461 self.base.base.shared_params()
462 }
463
464 fn import_params(&self) -> Option<&ImportParams> {
465 self.base.base.import_params()
466 }
467
468 fn network_params(&self) -> Option<&NetworkParams> {
469 self.base.base.network_params()
470 }
471
472 fn keystore_params(&self) -> Option<&KeystoreParams> {
473 self.base.base.keystore_params()
474 }
475
476 fn base_path(&self) -> sc_cli::Result<Option<BasePath>> {
477 Ok(self
478 .shared_params()
479 .base_path()?
480 .or_else(|| self.base_path.clone().map(Into::into)))
481 }
482
483 fn rpc_addr(&self, default_listen_port: u16) -> sc_cli::Result<Option<Vec<RpcEndpoint>>> {
484 self.base.base.rpc_addr(default_listen_port)
485 }
486
487 fn prometheus_config(
488 &self,
489 default_listen_port: u16,
490 chain_spec: &Box<dyn ChainSpec>,
491 ) -> sc_cli::Result<Option<PrometheusConfig>> {
492 self.base.base.prometheus_config(default_listen_port, chain_spec)
493 }
494
495 fn init<F>(
496 &self,
497 _support_url: &String,
498 _impl_version: &String,
499 _logger_hook: F,
500 ) -> sc_cli::Result<()>
501 where
502 F: FnOnce(&mut sc_cli::LoggerBuilder),
503 {
504 unreachable!("PolkadotCli is never initialized; qed");
505 }
506
507 fn chain_id(&self, is_dev: bool) -> sc_cli::Result<String> {
508 let chain_id = self.base.base.chain_id(is_dev)?;
509
510 Ok(if chain_id.is_empty() { self.chain_id.clone().unwrap_or_default() } else { chain_id })
511 }
512
513 fn role(&self, is_dev: bool) -> sc_cli::Result<sc_service::Role> {
514 self.base.base.role(is_dev)
515 }
516
517 fn transaction_pool(
518 &self,
519 is_dev: bool,
520 ) -> sc_cli::Result<sc_service::config::TransactionPoolOptions> {
521 self.base.base.transaction_pool(is_dev)
522 }
523
524 fn trie_cache_maximum_size(&self) -> sc_cli::Result<Option<usize>> {
525 self.base.base.trie_cache_maximum_size()
526 }
527
528 fn rpc_methods(&self) -> sc_cli::Result<sc_service::config::RpcMethods> {
529 self.base.base.rpc_methods()
530 }
531
532 fn rpc_max_connections(&self) -> sc_cli::Result<u32> {
533 self.base.base.rpc_max_connections()
534 }
535
536 fn rpc_cors(&self, is_dev: bool) -> sc_cli::Result<Option<Vec<String>>> {
537 self.base.base.rpc_cors(is_dev)
538 }
539
540 fn default_heap_pages(&self) -> sc_cli::Result<Option<u64>> {
541 self.base.base.default_heap_pages()
542 }
543
544 fn force_authoring(&self) -> sc_cli::Result<bool> {
545 self.base.base.force_authoring()
546 }
547
548 fn disable_grandpa(&self) -> sc_cli::Result<bool> {
549 self.base.base.disable_grandpa()
550 }
551
552 fn max_runtime_instances(&self) -> sc_cli::Result<Option<usize>> {
553 self.base.base.max_runtime_instances()
554 }
555
556 fn announce_block(&self) -> sc_cli::Result<bool> {
557 self.base.base.announce_block()
558 }
559
560 fn telemetry_endpoints(
561 &self,
562 chain_spec: &Box<dyn ChainSpec>,
563 ) -> sc_cli::Result<Option<sc_telemetry::TelemetryEndpoints>> {
564 self.base.base.telemetry_endpoints(chain_spec)
565 }
566
567 fn node_name(&self) -> sc_cli::Result<String> {
568 self.base.base.node_name()
569 }
570}