From 9290378b3bc84bf152d0cdac945aa31bb2bdf9d5 Mon Sep 17 00:00:00 2001 From: Trantorian1 <114066155+Trantorian1@users.noreply.github.com> Date: Wed, 8 Jan 2025 11:47:38 +0100 Subject: [PATCH] fix(devnet): loosened devnet chain id restrictions (#448) --- README.md | 20 ++++++++++------- .../madara/node/src/cli/block_production.rs | 5 ----- crates/madara/node/src/cli/mod.rs | 16 ++++---------- crates/madara/node/src/main.rs | 22 +++++++++---------- 4 files changed, 27 insertions(+), 36 deletions(-) diff --git a/README.md b/README.md index 47906d2ff..a6330d505 100644 --- a/README.md +++ b/README.md @@ -94,6 +94,10 @@ You can choose between different build modes: Start the Madara client with a basic set of arguments depending on your chosen mode: +> [!NOTE] +> Head to the [Configuration](#%EF%B8%8F-configuration) section to learn more about +> customizing your node. + #### Full Node Synchronizes the state of the chain from genesis. @@ -125,16 +129,16 @@ cargo run --release -- \ A node in a private local network. ```bash -cargo run --release -- \ - --name Madara \ - --devnet \ - --base-path /var/lib/madara \ - --preset sepolia + cargo run --release -- \ + --name Madara \ + --devnet \ + --base-path ../madara_db \ + --chain-config-override=chain_id=MY_CUSTOM_DEVNET ``` -> [!NOTE] -> Head to the [Configuration](#%EF%B8%8F-configuration) section to learn more about -> customizing your node. +> [!CAUTION] +> Make sure to use a unique `chain_id` for your devnet to avoid potential replay +> attacks in other chains with the same chain id! #### 4. Presets diff --git a/crates/madara/node/src/cli/block_production.rs b/crates/madara/node/src/cli/block_production.rs index e85d18a10..ee4dd097a 100644 --- a/crates/madara/node/src/cli/block_production.rs +++ b/crates/madara/node/src/cli/block_production.rs @@ -6,11 +6,6 @@ pub struct BlockProductionParams { #[arg(env = "MADARA_BLOCK_PRODUCTION_DISABLED", long, alias = "no-block-production")] pub block_production_disabled: bool, - /// Launch a devnet with a production chain id (like SN_MAINNET, SN_SEPOLIA). - /// This in unsafe because your devnet transactions can be replayed on the actual network. - #[arg(env = "MADARA_OVERRIDE_DEVNET_CHAIN_ID", long, default_value_t = false)] - pub override_devnet_chain_id: bool, - /// Create this number of contracts in the genesis block for the devnet configuration. #[arg(env = "MADARA_DEVNET_CONTRACTS", long, default_value_t = 10)] pub devnet_contracts: u64, diff --git a/crates/madara/node/src/cli/mod.rs b/crates/madara/node/src/cli/mod.rs index 557f6d8de..9fae86aa8 100644 --- a/crates/madara/node/src/cli/mod.rs +++ b/crates/madara/node/src/cli/mod.rs @@ -16,7 +16,6 @@ pub use db::*; pub use gateway::*; pub use l2::*; pub use rpc::*; -use starknet_api::core::ChainId; use std::str::FromStr; pub use telemetry::*; @@ -188,6 +187,10 @@ pub struct RunCmd { #[arg(env = "MADARA_DEVNET", long, group = "mode")] pub devnet: bool, + /// Allows a devnet to have SN_MAIN or SN_SEPOLIA as its chain ID. + #[arg(env = "MADARA_DEVNET_UNSAFE", long, requires = "devnet")] + pub devnet_unsafe: bool, + /// The network chain configuration. #[clap(env = "MADARA_NETWORK", long, short, group = "full_mode_config")] pub network: Option, @@ -313,17 +316,6 @@ pub enum NetworkType { Devnet, } -impl NetworkType { - pub fn chain_id(&self) -> ChainId { - match self { - NetworkType::Main => ChainId::Mainnet, - NetworkType::Test => ChainId::Sepolia, - NetworkType::Integration => ChainId::IntegrationSepolia, - NetworkType::Devnet => ChainId::Other("MADARA_DEVNET".to_string()), - } - } -} - #[derive(Debug, Clone, clap::ValueEnum)] #[value(rename_all = "kebab-case")] pub enum ChainPreset { diff --git a/crates/madara/node/src/main.rs b/crates/madara/node/src/main.rs index b0958cc46..00314928b 100644 --- a/crates/madara/node/src/main.rs +++ b/crates/madara/node/src/main.rs @@ -7,7 +7,7 @@ mod util; use anyhow::{bail, Context}; use clap::Parser; -use cli::{NetworkType, RunCmd}; +use cli::RunCmd; use http::{HeaderName, HeaderValue}; use mc_analytics::Analytics; use mc_block_import::BlockImporter; @@ -20,6 +20,7 @@ use mc_telemetry::{SysInfo, TelemetryService}; use mp_oracle::pragma::PragmaOracleBuilder; use mp_utils::service::{MadaraServiceId, ServiceMonitor}; use service::{BlockProductionService, GatewayService, L1SyncService, L2SyncService, RpcService}; +use starknet_api::core::ChainId; use std::sync::Arc; const GREET_IMPL_NAME: &str = "Madara"; @@ -50,16 +51,15 @@ async fn main() -> anyhow::Result<()> { run_cmd.chain_config()? }; - // Check if the devnet is running with the correct chain id. - if run_cmd.devnet && chain_config.chain_id != NetworkType::Devnet.chain_id() { - if !run_cmd.block_production_params.override_devnet_chain_id { - tracing::error!("You're running a devnet with the network config of {:?}. This means that devnet transactions can be replayed on the actual network. Use `--network=devnet` instead. Or if this is the expected behavior please pass `--override-devnet-chain-id`", chain_config.chain_name); - panic!(); - } else { - // This log is immediately flooded with devnet accounts and so this can be missed. - // Should we add a delay here to make this clearly visisble? - tracing::warn!("You're running a devnet with the network config of {:?}. This means that devnet transactions can be replayed on the actual network.", run_cmd.network); - } + // Check if the devnet is running with the correct chain id. This is purely + // to avoid accidental setups which would allow for replay attacks. This is + // possible if the devnet has the same chain id as another popular chain, + // allowing txs which occur on it to also be replayed on that other chain. + if run_cmd.devnet + && (chain_config.chain_id == ChainId::Mainnet || chain_config.chain_id == ChainId::Sepolia) + && !run_cmd.devnet_unsafe + { + anyhow::bail!("You're running a devnet with the network config of {0}. This means that devnet transactions can be replayed on the actual {0} network. Use `--network=devnet` instead or force this configuration with `--devnet-unsafe`.", chain_config.chain_name); } let node_name = run_cmd.node_name_or_provide().await.to_string();