From 2b47156499f98a8e0689d9654cea85af74777637 Mon Sep 17 00:00:00 2001 From: Ash Manning Date: Mon, 8 Jul 2024 20:14:44 +0800 Subject: [PATCH] add network option, fix bug when submitting blocks, bump version --- Cargo.lock | 35 ++++++++++++++++++++++++----- Cargo.toml | 2 +- app/Cargo.toml | 2 +- app/app.rs | 1 + app/cli.rs | 7 +++++- lib/Cargo.toml | 4 +++- lib/archive.rs | 58 ++++++++++++++++++++++++++++++++++++++++++++++++ lib/net/mod.rs | 28 ++++++++++++++++------- lib/node/mod.rs | 6 +++-- lib/types/mod.rs | 8 +++++++ 10 files changed, 131 insertions(+), 20 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 1142480..69288b0 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -3839,7 +3839,7 @@ checksum = "d231b230927b5e4ad203db57bbcbee2802f6bce620b1e4a9024a07d94e2907ec" [[package]] name = "plain_bitassets" -version = "0.9.1" +version = "0.9.2" dependencies = [ "addr", "anyhow", @@ -3876,6 +3876,7 @@ dependencies = [ "serde_json", "serde_with", "sha256", + "strum 0.26.3", "thiserror", "tiny-bip39", "tokio", @@ -3888,7 +3889,7 @@ dependencies = [ [[package]] name = "plain_bitassets_app" -version = "0.9.1" +version = "0.9.2" dependencies = [ "anyhow", "async_zmq", @@ -3917,7 +3918,7 @@ dependencies = [ "poll-promise", "serde", "shlex", - "strum", + "strum 0.25.0", "tempfile", "test-log", "thiserror", @@ -3932,7 +3933,7 @@ dependencies = [ [[package]] name = "plain_bitassets_app_cli" -version = "0.9.1" +version = "0.9.2" dependencies = [ "anyhow", "bip300301", @@ -3947,7 +3948,7 @@ dependencies = [ [[package]] name = "plain_bitassets_app_rpc_api" -version = "0.9.1" +version = "0.9.2" dependencies = [ "bip300301", "fraction", @@ -4936,7 +4937,16 @@ version = "0.25.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "290d54ea6f91c969195bdbcd7442c8c2a2ba87da8bf60a7ee86a235d4bc1e125" dependencies = [ - "strum_macros", + "strum_macros 0.25.3", +] + +[[package]] +name = "strum" +version = "0.26.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8fec0f0aef304996cf250b31b5a10dee7980c85da9d759361292b8bca5a18f06" +dependencies = [ + "strum_macros 0.26.4", ] [[package]] @@ -4952,6 +4962,19 @@ dependencies = [ "syn 2.0.68", ] +[[package]] +name = "strum_macros" +version = "0.26.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4c6bee85a5a24955dc440386795aa378cd9cf82acd5f764469152d2270e581be" +dependencies = [ + "heck 0.5.0", + "proc-macro2", + "quote", + "rustversion", + "syn 2.0.68", +] + [[package]] name = "subtle" version = "2.6.0" diff --git a/Cargo.toml b/Cargo.toml index 7a63cf4..786a370 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -10,7 +10,7 @@ members = [ [workspace.package] authors = [ "Ash Manning " ] edition = "2021" -version = "0.9.1" +version = "0.9.2" [workspace.dependencies.bip300301] git = "https://github.com/Ash-L2L/bip300301.git" diff --git a/app/Cargo.toml b/app/Cargo.toml index 544d925..e363b08 100644 --- a/app/Cargo.toml +++ b/app/Cargo.toml @@ -31,7 +31,7 @@ itertools = "0.11.0" include_path = "0.1.1" jsonrpsee = { version = "0.20.0", features = ["server"] } parking_lot = "0.12.1" -plain_bitassets = { path = "../lib" } +plain_bitassets = { path = "../lib", features = ["clap"] } plain_bitassets_app_cli = { path = "../cli" } plain_bitassets_app_rpc_api = { path = "../rpc-api" } poll-promise = { version = "0.3.0", features = ["tokio"] } diff --git a/app/app.rs b/app/app.rs index 654acdb..aef38a4 100644 --- a/app/app.rs +++ b/app/app.rs @@ -143,6 +143,7 @@ impl App { config.net_addr, &config.datadir, config.main_addr, + config.network, &config.main_password, &config.main_user, local_pool.clone(), diff --git a/app/cli.rs b/app/cli.rs index 9388364..ae7414e 100644 --- a/app/cli.rs +++ b/app/cli.rs @@ -6,7 +6,7 @@ use std::{ }; use clap::{Arg, Parser}; -use plain_bitassets::node::THIS_SIDECHAIN; +use plain_bitassets::{node::THIS_SIDECHAIN, types::Network}; const fn ipv4_socket_addr(ipv4_octets: [u8; 4], port: u16) -> SocketAddr { let [a, b, c, d] = ipv4_octets; @@ -131,6 +131,9 @@ pub(super) struct Cli { /// Socket address to use for P2P networking #[arg(default_value_t = DEFAULT_NET_ADDR, long, short)] net_addr: SocketAddr, + /// Set the network. Setting this may affect other defaults. + #[arg(default_value_t, long, value_enum)] + network: Network, /// Socket address to host the RPC server #[arg(default_value_t = DEFAULT_RPC_ADDR, long, short)] rpc_addr: SocketAddr, @@ -158,6 +161,7 @@ pub struct Config { pub mnemonic_seed_phrase_path: Option, pub main_user: String, pub net_addr: SocketAddr, + pub network: Network, pub rpc_addr: SocketAddr, #[cfg(all(not(target_os = "windows"), feature = "zmq"))] pub zmq_addr: SocketAddr, @@ -191,6 +195,7 @@ impl Cli { main_user: self.user_main, mnemonic_seed_phrase_path: self.mnemonic_seed_phrase_path, net_addr: self.net_addr, + network: self.network, rpc_addr: self.rpc_addr, #[cfg(all(not(target_os = "windows"), feature = "zmq"))] zmq_addr: self.zmq_addr, diff --git a/lib/Cargo.toml b/lib/Cargo.toml index beaac80..fafa939 100644 --- a/lib/Cargo.toml +++ b/lib/Cargo.toml @@ -15,7 +15,7 @@ borsh = {version = "1.3.0", features = ["derive"] } bs58 = { version = "0.5.0", features = ["check"] } byteorder = "1.4.3" bytes = "1.4.0" -clap = { version = "4.5.4", features = ["derive"] } +clap = { version = "4.5.4", features = ["derive"], optional = true } ed25519-dalek = { version = "2.1.1", features = ["batch", "serde"] } ed25519-dalek-bip32 = "0.3.0" educe = { version = "0.4.23", features = ["Hash"] } @@ -39,6 +39,7 @@ serde = { version = "1.0.179", features = ["derive"] } serde_json = "1.0.113" serde_with = { version = "3.4.0", default-features = false } sha256 = "1.2.2" +strum = { version = "0.26.3", features = ["derive"], optional = true } thiserror = "1.0.44" tiny-bip39 = "1.0.0" tokio = { version = "1.29.1", features = ["rt-multi-thread", "sync"] } @@ -54,6 +55,7 @@ rev = "61748742c2a91be4be24a227d109fd93492f7484" optional = true [features] +clap = ["dep:clap", "dep:strum"] zmq = ["dep:async_zmq"] [lib] diff --git a/lib/archive.rs b/lib/archive.rs index c9b150a..e2c42eb 100644 --- a/lib/archive.rs +++ b/lib/archive.rs @@ -723,6 +723,64 @@ impl Archive { &block_hash, &exponential_ancestors, )?; + // Populate BMM verifications + { + let mut bmm_results = self.get_bmm_results(rwtxn, block_hash)?; + let parent_bmm_results = + self.get_bmm_results(rwtxn, header.prev_side_hash)?; + let main_blocks = + self.get_main_successors(rwtxn, header.prev_main_hash)?; + for main_block in main_blocks { + let Some(commitment) = + self.get_main_bmm_commitment(rwtxn, main_block)? + else { + tracing::trace!(%block_hash, "Failed BMM @ {main_block}: missing commitment"); + bmm_results.insert(main_block, BmmResult::Failed); + continue; + }; + if commitment != block_hash { + tracing::trace!(%block_hash, "Failed BMM @ {main_block}: commitment to other block ({commitment})"); + bmm_results.insert(main_block, BmmResult::Failed); + continue; + } + let main_header = self.get_main_header(rwtxn, main_block)?; + if header.prev_main_hash != main_header.prev_blockhash { + tracing::trace!(%block_hash, "Failed BMM @ {main_block}: should be impossible?"); + bmm_results.insert(main_block, BmmResult::Failed); + continue; + } + if header.prev_side_hash == BlockHash::default() { + tracing::trace!(%block_hash, "Verified BMM @ {main_block}: no parent"); + bmm_results.insert(main_block, BmmResult::Verified); + continue; + } + // Check if there is a valid BMM commitment to the parent in the + // main ancestry + let main_ancestry_contains_valid_bmm_commitment_to_parent = + parent_bmm_results + .iter() + .map(Ok) + .transpose_into_fallible() + .any(|(bmm_block, bmm_result)| { + let parent_verified = *bmm_result + == BmmResult::Verified + && self.is_main_descendant( + rwtxn, *bmm_block, main_block, + )?; + Result::::Ok(parent_verified) + })?; + if main_ancestry_contains_valid_bmm_commitment_to_parent { + tracing::trace!(%block_hash, "Verified BMM @ {main_block}: verified parent"); + bmm_results.insert(main_block, BmmResult::Verified); + continue; + } else { + tracing::trace!(%block_hash, "Failed BMM @ {main_block}: no valid BMM commitment to parent in main ancestry"); + bmm_results.insert(main_block, BmmResult::Failed); + continue; + } + } + self.bmm_results.put(rwtxn, &block_hash, &bmm_results)?; + } Ok(()) } diff --git a/lib/net/mod.rs b/lib/net/mod.rs index c0cb801..e5c4e50 100644 --- a/lib/net/mod.rs +++ b/lib/net/mod.rs @@ -16,7 +16,7 @@ use tokio_stream::StreamNotifyClose; use crate::{ archive::Archive, node::THIS_SIDECHAIN, state::State, - types::AuthorizedTransaction, + types::{AuthorizedTransaction, Network} }; mod peer; @@ -78,6 +78,21 @@ pub fn make_client_endpoint(bind_addr: SocketAddr) -> Result { pub type PeerInfoRx = mpsc::UnboundedReceiver<(SocketAddr, Option)>; +const SIGNET_SEED_NODE_ADDRS: &[SocketAddr] = { + const SEED_NODE_ADDR: SocketAddr = SocketAddr::new( + std::net::IpAddr::V4(std::net::Ipv4Addr::new(172, 105, 148, 135)), + 4000 + THIS_SIDECHAIN as u16, + ); + &[SEED_NODE_ADDR] +}; + +const fn seed_node_addrs(network: Network) -> &'static [SocketAddr] { + match network { + Network::Signet => SIGNET_SEED_NODE_ADDRS, + Network::Regtest => &[], + } +} + // State. // Archive. @@ -165,6 +180,7 @@ impl Net { pub fn new( env: &heed::Env, archive: Archive, + network: Network, state: State, bind_addr: SocketAddr, ) -> Result<(Self, PeerInfoRx), Error> { @@ -178,13 +194,9 @@ impl Net { None => { let known_peers = env.create_database(&mut rwtxn, Some("known_peers"))?; - const SEED_NODE_ADDR: SocketAddr = SocketAddr::new( - std::net::IpAddr::V4(std::net::Ipv4Addr::new( - 172, 105, 148, 135, - )), - 4000 + THIS_SIDECHAIN as u16, - ); - known_peers.put(&mut rwtxn, &SEED_NODE_ADDR, &())?; + for seed_node_addr in seed_node_addrs(network) { + known_peers.put(&mut rwtxn, seed_node_addr, &())?; + } known_peers } }; diff --git a/lib/node/mod.rs b/lib/node/mod.rs index a4e8ce8..8e19673 100644 --- a/lib/node/mod.rs +++ b/lib/node/mod.rs @@ -25,7 +25,7 @@ use crate::{ Address, AssetId, Authorized, AuthorizedTransaction, BitAssetData, BitAssetId, Block, BlockHash, BmmResult, Body, DutchAuctionId, FilledOutput, FilledTransaction, GetBitcoinValue, Header, InPoint, - OutPoint, Output, SpentOutput, Tip, Transaction, TxIn, Txid, + Network, OutPoint, Output, SpentOutput, Tip, Transaction, TxIn, Txid, WithdrawalBundle, }, util::Watchable, @@ -157,10 +157,12 @@ pub struct Node { } impl Node { + #[allow(clippy::too_many_arguments)] pub fn new( bind_addr: SocketAddr, datadir: &Path, main_addr: SocketAddr, + network: Network, password: &str, user: &str, local_pool: LocalPoolHandle, @@ -199,7 +201,7 @@ impl Node { drivechain.clone(), ); let (net, peer_info_rx) = - Net::new(&env, archive.clone(), state.clone(), bind_addr)?; + Net::new(&env, archive.clone(), network, state.clone(), bind_addr)?; let net_task = NetTaskHandle::new( local_pool.clone(), diff --git a/lib/types/mod.rs b/lib/types/mod.rs index 2cf4d06..b6434c9 100644 --- a/lib/types/mod.rs +++ b/lib/types/mod.rs @@ -436,6 +436,14 @@ impl Default for Tip { } } +#[derive(Clone, Copy, Debug, Default, Eq, Hash, PartialEq)] +#[cfg_attr(feature = "clap", derive(clap::ValueEnum, strum::Display))] +pub enum Network { + #[default] + Signet, + Regtest, +} + pub mod open_api_schemas { pub use super::output::open_api_schemas::*; pub use super::transaction::open_api_schemas::*;