From 1c2f7660ec73da2d69a88846d5c9d3495828d4a2 Mon Sep 17 00:00:00 2001 From: Michael Sutton Date: Thu, 28 Sep 2023 14:19:29 +0300 Subject: [PATCH] Apply mempool dust prevention patch + version prep (#286) * reject spam transactions * allow multiple grpc client connections from the same ip * log utxo set size * improve spam logs * rollback virtual utxoset sweep on startup * block dust txs only on mainnet * bump version to 0.1.7 * fine tune DB file limits for non-consensus DBs --- Cargo.lock | 99 ++++++++++--------- Cargo.toml | 94 +++++++++--------- consensus/core/src/network.rs | 4 + kaspad/src/daemon.rs | 22 ++++- mining/errors/src/mempool.rs | 3 + mining/src/manager.rs | 16 +++ mining/src/mempool/config.rs | 17 +++- .../validate_and_insert_transaction.rs | 17 +++- protocol/flows/src/v5/txrelay/flow.rs | 20 +++- rpc/grpc/server/Cargo.toml | 1 + rpc/grpc/server/src/connection.rs | 16 ++- rpc/grpc/server/src/manager.rs | 8 +- 12 files changed, 201 insertions(+), 116 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 8f81a1a50..8dd043287 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -1955,7 +1955,7 @@ dependencies = [ [[package]] name = "kaspa-addresses" -version = "0.1.6" +version = "0.1.7" dependencies = [ "borsh", "criterion", @@ -1971,7 +1971,7 @@ dependencies = [ [[package]] name = "kaspa-addressmanager" -version = "0.1.6" +version = "0.1.7" dependencies = [ "borsh", "itertools 0.10.5", @@ -1991,7 +1991,7 @@ dependencies = [ [[package]] name = "kaspa-bip32" -version = "0.1.6" +version = "0.1.7" dependencies = [ "bs58 0.4.0", "faster-hex 0.6.1", @@ -2015,7 +2015,7 @@ dependencies = [ [[package]] name = "kaspa-cli" -version = "0.1.6" +version = "0.1.7" dependencies = [ "async-trait", "borsh", @@ -2058,7 +2058,7 @@ dependencies = [ [[package]] name = "kaspa-connectionmanager" -version = "0.1.6" +version = "0.1.7" dependencies = [ "duration-string", "futures-util", @@ -2075,7 +2075,7 @@ dependencies = [ [[package]] name = "kaspa-consensus" -version = "0.1.6" +version = "0.1.7" dependencies = [ "async-channel", "bincode", @@ -2117,7 +2117,7 @@ dependencies = [ [[package]] name = "kaspa-consensus-core" -version = "0.1.6" +version = "0.1.7" dependencies = [ "async-trait", "bincode", @@ -2154,7 +2154,7 @@ dependencies = [ [[package]] name = "kaspa-consensus-notify" -version = "0.1.6" +version = "0.1.7" dependencies = [ "async-channel", "cfg-if 1.0.0", @@ -2173,7 +2173,7 @@ dependencies = [ [[package]] name = "kaspa-consensus-wasm" -version = "0.1.6" +version = "0.1.7" dependencies = [ "faster-hex 0.6.1", "js-sys", @@ -2195,7 +2195,7 @@ dependencies = [ [[package]] name = "kaspa-consensusmanager" -version = "0.1.6" +version = "0.1.7" dependencies = [ "duration-string", "futures", @@ -2213,7 +2213,7 @@ dependencies = [ [[package]] name = "kaspa-core" -version = "0.1.6" +version = "0.1.7" dependencies = [ "cfg-if 1.0.0", "ctrlc", @@ -2231,7 +2231,7 @@ dependencies = [ [[package]] name = "kaspa-daemon" -version = "0.1.6" +version = "0.1.7" dependencies = [ "async-trait", "borsh", @@ -2253,7 +2253,7 @@ dependencies = [ [[package]] name = "kaspa-database" -version = "0.1.6" +version = "0.1.7" dependencies = [ "bincode", "enum-primitive-derive", @@ -2276,7 +2276,7 @@ dependencies = [ [[package]] name = "kaspa-grpc-client" -version = "0.1.6" +version = "0.1.7" dependencies = [ "async-channel", "async-stream", @@ -2303,7 +2303,7 @@ dependencies = [ [[package]] name = "kaspa-grpc-core" -version = "0.1.6" +version = "0.1.7" dependencies = [ "async-channel", "async-stream", @@ -2331,7 +2331,7 @@ dependencies = [ [[package]] name = "kaspa-grpc-server" -version = "0.1.6" +version = "0.1.7" dependencies = [ "async-channel", "async-stream", @@ -2357,11 +2357,12 @@ dependencies = [ "tokio-stream", "tonic", "triggered", + "uuid 1.4.1", ] [[package]] name = "kaspa-hashes" -version = "0.1.6" +version = "0.1.7" dependencies = [ "blake2b_simd", "borsh", @@ -2382,7 +2383,7 @@ dependencies = [ [[package]] name = "kaspa-index-core" -version = "0.1.6" +version = "0.1.7" dependencies = [ "async-channel", "async-trait", @@ -2401,7 +2402,7 @@ dependencies = [ [[package]] name = "kaspa-index-processor" -version = "0.1.6" +version = "0.1.7" dependencies = [ "async-channel", "async-trait", @@ -2429,7 +2430,7 @@ dependencies = [ [[package]] name = "kaspa-math" -version = "0.1.6" +version = "0.1.7" dependencies = [ "borsh", "criterion", @@ -2450,14 +2451,14 @@ dependencies = [ [[package]] name = "kaspa-merkle" -version = "0.1.6" +version = "0.1.7" dependencies = [ "kaspa-hashes", ] [[package]] name = "kaspa-mining" -version = "0.1.6" +version = "0.1.7" dependencies = [ "criterion", "futures-util", @@ -2481,7 +2482,7 @@ dependencies = [ [[package]] name = "kaspa-mining-errors" -version = "0.1.6" +version = "0.1.7" dependencies = [ "kaspa-consensus-core", "thiserror", @@ -2489,7 +2490,7 @@ dependencies = [ [[package]] name = "kaspa-muhash" -version = "0.1.6" +version = "0.1.7" dependencies = [ "criterion", "kaspa-hashes", @@ -2502,7 +2503,7 @@ dependencies = [ [[package]] name = "kaspa-notify" -version = "0.1.6" +version = "0.1.7" dependencies = [ "async-channel", "async-trait", @@ -2563,7 +2564,7 @@ dependencies = [ [[package]] name = "kaspa-p2p-flows" -version = "0.1.6" +version = "0.1.7" dependencies = [ "async-trait", "futures", @@ -2592,7 +2593,7 @@ dependencies = [ [[package]] name = "kaspa-p2p-lib" -version = "0.1.6" +version = "0.1.7" dependencies = [ "borsh", "ctrlc", @@ -2621,7 +2622,7 @@ dependencies = [ [[package]] name = "kaspa-perf-monitor" -version = "0.1.6" +version = "0.1.7" dependencies = [ "kaspa-core", "log", @@ -2633,7 +2634,7 @@ dependencies = [ [[package]] name = "kaspa-pow" -version = "0.1.6" +version = "0.1.7" dependencies = [ "criterion", "js-sys", @@ -2647,7 +2648,7 @@ dependencies = [ [[package]] name = "kaspa-rpc-core" -version = "0.1.6" +version = "0.1.7" dependencies = [ "async-channel", "async-trait", @@ -2681,7 +2682,7 @@ dependencies = [ [[package]] name = "kaspa-rpc-macros" -version = "0.1.6" +version = "0.1.7" dependencies = [ "convert_case 0.5.0", "proc-macro-error", @@ -2693,7 +2694,7 @@ dependencies = [ [[package]] name = "kaspa-rpc-service" -version = "0.1.6" +version = "0.1.7" dependencies = [ "async-trait", "kaspa-addresses", @@ -2720,7 +2721,7 @@ dependencies = [ [[package]] name = "kaspa-testing-integration" -version = "0.1.6" +version = "0.1.7" dependencies = [ "async-channel", "bincode", @@ -2771,7 +2772,7 @@ dependencies = [ [[package]] name = "kaspa-txscript" -version = "0.1.6" +version = "0.1.7" dependencies = [ "blake2b_simd", "borsh", @@ -2796,7 +2797,7 @@ dependencies = [ [[package]] name = "kaspa-txscript-errors" -version = "0.1.6" +version = "0.1.7" dependencies = [ "secp256k1", "thiserror", @@ -2804,7 +2805,7 @@ dependencies = [ [[package]] name = "kaspa-utils" -version = "0.1.6" +version = "0.1.7" dependencies = [ "async-channel", "async-trait", @@ -2828,7 +2829,7 @@ dependencies = [ [[package]] name = "kaspa-utxoindex" -version = "0.1.6" +version = "0.1.7" dependencies = [ "futures", "kaspa-consensus", @@ -2849,7 +2850,7 @@ dependencies = [ [[package]] name = "kaspa-wallet" -version = "0.1.6" +version = "0.1.7" dependencies = [ "async-std", "async-trait", @@ -2861,7 +2862,7 @@ dependencies = [ [[package]] name = "kaspa-wallet-cli-wasm" -version = "0.1.6" +version = "0.1.7" dependencies = [ "async-trait", "js-sys", @@ -2875,7 +2876,7 @@ dependencies = [ [[package]] name = "kaspa-wallet-core" -version = "0.1.6" +version = "0.1.7" dependencies = [ "aes", "argon2", @@ -2939,7 +2940,7 @@ dependencies = [ [[package]] name = "kaspa-wasm" -version = "0.1.6" +version = "0.1.7" dependencies = [ "js-sys", "kaspa-addresses", @@ -2959,7 +2960,7 @@ dependencies = [ [[package]] name = "kaspa-wrpc-client" -version = "0.1.6" +version = "0.1.7" dependencies = [ "async-std", "async-trait", @@ -2989,11 +2990,11 @@ dependencies = [ [[package]] name = "kaspa-wrpc-core" -version = "0.1.6" +version = "0.1.7" [[package]] name = "kaspa-wrpc-proxy" -version = "0.1.6" +version = "0.1.7" dependencies = [ "async-trait", "clap 4.3.19", @@ -3013,7 +3014,7 @@ dependencies = [ [[package]] name = "kaspa-wrpc-server" -version = "0.1.6" +version = "0.1.7" dependencies = [ "async-trait", "borsh", @@ -3040,14 +3041,14 @@ dependencies = [ [[package]] name = "kaspa-wrpc-wasm" -version = "0.1.6" +version = "0.1.7" dependencies = [ "kaspa-wrpc-client", ] [[package]] name = "kaspad" -version = "0.1.6" +version = "0.1.7" dependencies = [ "async-channel", "clap 4.3.19", @@ -4263,7 +4264,7 @@ dependencies = [ [[package]] name = "rothschild" -version = "0.1.6" +version = "0.1.7" dependencies = [ "clap 4.3.19", "faster-hex 0.6.1", @@ -4645,7 +4646,7 @@ dependencies = [ [[package]] name = "simpa" -version = "0.1.6" +version = "0.1.7" dependencies = [ "async-channel", "clap 4.3.19", diff --git a/Cargo.toml b/Cargo.toml index d788e8ce5..bf3ace06f 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -54,7 +54,7 @@ members = [ ] [workspace.package] -version = "0.1.6" +version = "0.1.7" authors = ["Kaspa developers"] license = "MIT/Apache-2.0" edition = "2021" @@ -70,53 +70,53 @@ include = [ ] [workspace.dependencies] -# kaspa-testing-integration = { version = "0.1.2", path = "testing/integration" } -kaspa-os = { version = "0.1.6", path = "kaspa-os" } -kaspa-daemon = { version = "0.1.6", path = "daemon" } -kaspa-addresses = { version = "0.1.6", path = "crypto/addresses" } -kaspa-addressmanager = { version = "0.1.6", path = "components/addressmanager" } -kaspa-bip32 = { version = "0.1.6", path = "wallet/bip32" } -kaspa-connectionmanager = { version = "0.1.6", path = "components/connectionmanager" } -kaspa-consensus = { version = "0.1.6", path = "consensus" } -kaspa-consensus-core = { version = "0.1.6", path = "consensus/core" } -kaspa-consensus-notify = { version = "0.1.6", path = "consensus/notify" } -kaspa-consensus-wasm = { version = "0.1.6", path = "consensus/wasm" } -kaspa-consensusmanager = { version = "0.1.6", path = "components/consensusmanager" } -kaspa-core = { version = "0.1.6", path = "core" } -kaspa-database = { version = "0.1.6", path = "database" } -kaspa-grpc-client = { version = "0.1.6", path = "rpc/grpc/client" } -kaspa-grpc-core = { version = "0.1.6", path = "rpc/grpc/core" } -kaspa-grpc-server = { version = "0.1.6", path = "rpc/grpc/server" } -kaspa-hashes = { version = "0.1.6", path = "crypto/hashes" } -kaspa-index-core = { version = "0.1.6", path = "indexes/core" } -kaspa-index-processor = { version = "0.1.6", path = "indexes/processor" } -kaspa-math = { version = "0.1.6", path = "math" } -kaspa-merkle = { version = "0.1.6", path = "crypto/merkle" } -kaspa-mining = { version = "0.1.6", path = "mining" } +# kaspa-testing-integration = { version = "0.1.7", path = "testing/integration" } +kaspa-os = { version = "0.1.7", path = "kaspa-os" } +kaspa-daemon = { version = "0.1.7", path = "daemon" } +kaspa-addresses = { version = "0.1.7", path = "crypto/addresses" } +kaspa-addressmanager = { version = "0.1.7", path = "components/addressmanager" } +kaspa-bip32 = { version = "0.1.7", path = "wallet/bip32" } +kaspa-connectionmanager = { version = "0.1.7", path = "components/connectionmanager" } +kaspa-consensus = { version = "0.1.7", path = "consensus" } +kaspa-consensus-core = { version = "0.1.7", path = "consensus/core" } +kaspa-consensus-notify = { version = "0.1.7", path = "consensus/notify" } +kaspa-consensus-wasm = { version = "0.1.7", path = "consensus/wasm" } +kaspa-consensusmanager = { version = "0.1.7", path = "components/consensusmanager" } +kaspa-core = { version = "0.1.7", path = "core" } +kaspa-database = { version = "0.1.7", path = "database" } +kaspa-grpc-client = { version = "0.1.7", path = "rpc/grpc/client" } +kaspa-grpc-core = { version = "0.1.7", path = "rpc/grpc/core" } +kaspa-grpc-server = { version = "0.1.7", path = "rpc/grpc/server" } +kaspa-hashes = { version = "0.1.7", path = "crypto/hashes" } +kaspa-index-core = { version = "0.1.7", path = "indexes/core" } +kaspa-index-processor = { version = "0.1.7", path = "indexes/processor" } +kaspa-math = { version = "0.1.7", path = "math" } +kaspa-merkle = { version = "0.1.7", path = "crypto/merkle" } +kaspa-mining = { version = "0.1.7", path = "mining" } kaspa-mining-errors = { path = "mining/errors" } -kaspa-muhash = { version = "0.1.6", path = "crypto/muhash" } -kaspa-notify = { version = "0.1.6", path = "notify" } -kaspa-p2p-flows = { version = "0.1.6", path = "protocol/flows" } -kaspa-p2p-lib = { version = "0.1.6", path = "protocol/p2p" } -kaspa-pow = { version = "0.1.6", path = "consensus/pow" } -kaspa-rpc-core = { version = "0.1.6", path = "rpc/core" } -kaspa-rpc-macros = { version = "0.1.6", path = "rpc/macros" } -kaspa-rpc-service = { version = "0.1.6", path = "rpc/service" } -kaspa-txscript = { version = "0.1.6", path = "crypto/txscript" } -kaspa-txscript-errors = { version = "0.1.6", path = "crypto/txscript/errors" } -kaspa-utils = { version = "0.1.6", path = "utils" } -kaspa-utxoindex = { version = "0.1.6", path = "indexes/utxoindex" } -kaspa-wallet = { version = "0.1.6", path = "wallet/native" } -kaspa-cli = { version = "0.1.6", path = "cli" } -kaspa-wallet-cli-wasm = { version = "0.1.6", path = "wallet/wasm" } -kaspa-wallet-core = { version = "0.1.6", path = "wallet/core" } -kaspa-wasm = { version = "0.1.6", path = "wasm" } -kaspa-wrpc-core = { version = "0.1.6", path = "rpc/wrpc/core" } -kaspa-wrpc-client = { version = "0.1.6", path = "rpc/wrpc/client" } -kaspa-wrpc-proxy = { version = "0.1.6", path = "rpc/wrpc/proxy" } -kaspa-wrpc-server = { version = "0.1.6", path = "rpc/wrpc/server" } -kaspa-wrpc-wasm = { version = "0.1.6", path = "rpc/wrpc/wasm" } -kaspad = { version = "0.1.6", path = "kaspad" } +kaspa-muhash = { version = "0.1.7", path = "crypto/muhash" } +kaspa-notify = { version = "0.1.7", path = "notify" } +kaspa-p2p-flows = { version = "0.1.7", path = "protocol/flows" } +kaspa-p2p-lib = { version = "0.1.7", path = "protocol/p2p" } +kaspa-pow = { version = "0.1.7", path = "consensus/pow" } +kaspa-rpc-core = { version = "0.1.7", path = "rpc/core" } +kaspa-rpc-macros = { version = "0.1.7", path = "rpc/macros" } +kaspa-rpc-service = { version = "0.1.7", path = "rpc/service" } +kaspa-txscript = { version = "0.1.7", path = "crypto/txscript" } +kaspa-txscript-errors = { version = "0.1.7", path = "crypto/txscript/errors" } +kaspa-utils = { version = "0.1.7", path = "utils" } +kaspa-utxoindex = { version = "0.1.7", path = "indexes/utxoindex" } +kaspa-wallet = { version = "0.1.7", path = "wallet/native" } +kaspa-cli = { version = "0.1.7", path = "cli" } +kaspa-wallet-cli-wasm = { version = "0.1.7", path = "wallet/wasm" } +kaspa-wallet-core = { version = "0.1.7", path = "wallet/core" } +kaspa-wasm = { version = "0.1.7", path = "wasm" } +kaspa-wrpc-core = { version = "0.1.7", path = "rpc/wrpc/core" } +kaspa-wrpc-client = { version = "0.1.7", path = "rpc/wrpc/client" } +kaspa-wrpc-proxy = { version = "0.1.7", path = "rpc/wrpc/proxy" } +kaspa-wrpc-server = { version = "0.1.7", path = "rpc/wrpc/server" } +kaspa-wrpc-wasm = { version = "0.1.7", path = "rpc/wrpc/wasm" } +kaspad = { version = "0.1.7", path = "kaspad" } kaspa-perf-monitor = { path = "metrics/perf_monitor" } # external diff --git a/consensus/core/src/network.rs b/consensus/core/src/network.rs index 2ce3f105d..7c4f7351a 100644 --- a/consensus/core/src/network.rs +++ b/consensus/core/src/network.rs @@ -191,6 +191,10 @@ impl NetworkId { self.network_type } + pub fn is_mainnet(&self) -> bool { + self.network_type == NetworkType::Mainnet + } + pub fn suffix(&self) -> Option { self.suffix } diff --git a/kaspad/src/daemon.rs b/kaspad/src/daemon.rs index bfb0118ea..101e7470f 100644 --- a/kaspad/src/daemon.rs +++ b/kaspad/src/daemon.rs @@ -31,6 +31,8 @@ const DEFAULT_DATA_DIR: &str = "datadir"; const CONSENSUS_DB: &str = "consensus"; const UTXOINDEX_DB: &str = "utxoindex"; const META_DB: &str = "meta"; +const UTXO_INDEX_DB_FILE_LIMIT: i32 = 100; +const META_DB_FILE_LIMIT: i32 = 5; const DEFAULT_LOG_DIR: &str = "logs"; fn get_home_dir() -> PathBuf { @@ -192,7 +194,8 @@ do you confirm? (answer y/n or pass --yes to the Kaspad command line to confirm } // DB used for addresses store and for multi-consensus management - let mut meta_db = kaspa_database::prelude::ConnBuilder::default().with_db_path(meta_db_dir.clone()).build(); + let mut meta_db = + kaspa_database::prelude::ConnBuilder::default().with_files_limit(META_DB_FILE_LIMIT).with_db_path(meta_db_dir.clone()).build(); // TEMP: upgrade from Alpha version or any version before this one if meta_db.get_pinned(b"multi-consensus-metadata-key").is_ok_and(|r| r.is_some()) { @@ -213,7 +216,8 @@ do you confirm? (answer y/n or pass --yes to the Kaspad command line to confirm fs::create_dir_all(utxoindex_db_dir.as_path()).unwrap(); // Reopen the DB - meta_db = kaspa_database::prelude::ConnBuilder::default().with_db_path(meta_db_dir).build(); + meta_db = + kaspa_database::prelude::ConnBuilder::default().with_files_limit(META_DB_FILE_LIMIT).with_db_path(meta_db_dir).build(); } let connect_peers = args.connect_peers.iter().map(|x| x.normalize(config.default_p2p_port())).collect::>(); @@ -266,7 +270,10 @@ do you confirm? (answer y/n or pass --yes to the Kaspad command line to confirm let notify_service = Arc::new(NotifyService::new(notification_root.clone(), notification_recv)); let index_service: Option> = if args.utxoindex { // Use only a single thread for none-consensus databases - let utxoindex_db = kaspa_database::prelude::ConnBuilder::default().with_db_path(utxoindex_db_dir).build(); + let utxoindex_db = kaspa_database::prelude::ConnBuilder::default() + .with_files_limit(UTXO_INDEX_DB_FILE_LIMIT) + .with_db_path(utxoindex_db_dir) + .build(); let utxoindex = UtxoIndexProxy::new(UtxoIndex::new(consensus_manager.clone(), utxoindex_db).unwrap()); let index_service = Arc::new(IndexService::new(¬ify_service.notifier(), Some(utxoindex))); Some(index_service) @@ -275,8 +282,13 @@ do you confirm? (answer y/n or pass --yes to the Kaspad command line to confirm }; let address_manager = AddressManager::new(config.clone(), meta_db); - let mining_manager = - MiningManagerProxy::new(Arc::new(MiningManager::new(config.target_time_per_block, false, config.max_block_mass, None))); + let mining_manager = MiningManagerProxy::new(Arc::new(MiningManager::new_with_spam_blocking_option( + network.is_mainnet(), + config.target_time_per_block, + false, + config.max_block_mass, + None, + ))); let flow_context = Arc::new(FlowContext::new( consensus_manager.clone(), diff --git a/mining/errors/src/mempool.rs b/mining/errors/src/mempool.rs index 0a4acbc79..6071caaf7 100644 --- a/mining/errors/src/mempool.rs +++ b/mining/errors/src/mempool.rs @@ -66,6 +66,9 @@ pub enum RuleError { // TODO: This error is added for the tx_relay flow but is never constructed neither in the golang nor in this version. Discuss if it can be removed. #[error("transaction {0} is invalid")] RejectInvalid(TransactionId), + + #[error("Rejected spam tx {0} from mempool")] + RejectSpamTransaction(TransactionId), } impl From for RuleError { diff --git a/mining/src/manager.rs b/mining/src/manager.rs index 4832bef4d..b8e448f55 100644 --- a/mining/src/manager.rs +++ b/mining/src/manager.rs @@ -44,6 +44,22 @@ impl MiningManager { Self::with_config(config, cache_lifetime) } + pub fn new_with_spam_blocking_option( + block_spam_txs: bool, + target_time_per_block: u64, + relay_non_std_transactions: bool, + max_block_mass: u64, + cache_lifetime: Option, + ) -> Self { + let config = Config::build_default_with_spam_blocking_option( + block_spam_txs, + target_time_per_block, + relay_non_std_transactions, + max_block_mass, + ); + Self::with_config(config, cache_lifetime) + } + pub(crate) fn with_config(config: Config, cache_lifetime: Option) -> Self { let block_template_builder = BlockTemplateBuilder::new(config.maximum_mass_per_block); let mempool = RwLock::new(Mempool::new(config)); diff --git a/mining/src/mempool/config.rs b/mining/src/mempool/config.rs index f567e8088..c238e4a67 100644 --- a/mining/src/mempool/config.rs +++ b/mining/src/mempool/config.rs @@ -38,6 +38,7 @@ pub struct Config { pub minimum_relay_transaction_fee: u64, pub minimum_standard_transaction_version: u16, pub maximum_standard_transaction_version: u16, + pub block_spam_txs: bool, } impl Config { @@ -56,6 +57,7 @@ impl Config { minimum_relay_transaction_fee: u64, minimum_standard_transaction_version: u16, maximum_standard_transaction_version: u16, + block_spam_txs: bool, ) -> Self { Self { maximum_transaction_count, @@ -71,12 +73,13 @@ impl Config { minimum_relay_transaction_fee, minimum_standard_transaction_version, maximum_standard_transaction_version, + block_spam_txs, } } /// Build a default config. /// The arguments should be obtained from the current consensus [`kaspa_consensus_core::config::params::Params`] instance. - pub fn build_default(target_milliseconds_per_block: u64, relay_non_std_transactions: bool, max_block_mass: u64) -> Self { + pub const fn build_default(target_milliseconds_per_block: u64, relay_non_std_transactions: bool, max_block_mass: u64) -> Self { Self { maximum_transaction_count: DEFAULT_MAXIMUM_TRANSACTION_COUNT, transaction_expire_interval_daa_score: DEFAULT_TRANSACTION_EXPIRE_INTERVAL_SECONDS * 1000 / target_milliseconds_per_block, @@ -92,6 +95,18 @@ impl Config { minimum_relay_transaction_fee: DEFAULT_MINIMUM_RELAY_TRANSACTION_FEE, minimum_standard_transaction_version: DEFAULT_MINIMUM_STANDARD_TRANSACTION_VERSION, maximum_standard_transaction_version: DEFAULT_MAXIMUM_STANDARD_TRANSACTION_VERSION, + block_spam_txs: false, } } + + /// Build a default config with optional spam blocking. + /// The arguments should be obtained from the current consensus [`kaspa_consensus_core::config::params::Params`] instance. + pub const fn build_default_with_spam_blocking_option( + block_spam_txs: bool, + target_milliseconds_per_block: u64, + relay_non_std_transactions: bool, + max_block_mass: u64, + ) -> Self { + Self { block_spam_txs, ..Self::build_default(target_milliseconds_per_block, relay_non_std_transactions, max_block_mass) } + } } diff --git a/mining/src/mempool/validate_and_insert_transaction.rs b/mining/src/mempool/validate_and_insert_transaction.rs index f4e915de6..27f4d17f9 100644 --- a/mining/src/mempool/validate_and_insert_transaction.rs +++ b/mining/src/mempool/validate_and_insert_transaction.rs @@ -7,7 +7,7 @@ use crate::mempool::{ }; use kaspa_consensus_core::{ api::ConsensusApi, - constants::UNACCEPTED_DAA_SCORE, + constants::{SOMPI_PER_KASPA, UNACCEPTED_DAA_SCORE}, tx::{MutableTransaction, Transaction, TransactionId, TransactionOutpoint, UtxoEntry}, }; use kaspa_core::info; @@ -84,6 +84,21 @@ impl Mempool { } fn validate_transaction_in_context(&self, transaction: &MutableTransaction) -> RuleResult<()> { + if self.config.block_spam_txs { + // TEMP: apply go-kaspad mempool dust prevention patch + // Note: we do not apply the part of the patch which modifies BBT since + // we do not support BBT on mainnet yet + let has_coinbase_input = transaction.entries.iter().any(|e| e.as_ref().unwrap().is_coinbase); + let num_extra_outs = transaction.tx.outputs.len() as i64 - transaction.tx.inputs.len() as i64; + if !has_coinbase_input + && num_extra_outs > 2 + && transaction.calculated_fee.unwrap() < num_extra_outs as u64 * SOMPI_PER_KASPA + { + kaspa_core::trace!("Rejected spam tx {} from mempool ({} outputs)", transaction.id(), transaction.tx.outputs.len()); + return Err(RuleError::RejectSpamTransaction(transaction.id())); + } + } + if !self.config.accept_non_standard { self.check_transaction_standard_in_context(transaction)?; } diff --git a/protocol/flows/src/v5/txrelay/flow.rs b/protocol/flows/src/v5/txrelay/flow.rs index ce9ba3c21..253cb1b2c 100644 --- a/protocol/flows/src/v5/txrelay/flow.rs +++ b/protocol/flows/src/v5/txrelay/flow.rs @@ -44,6 +44,9 @@ pub struct RelayTransactionsFlow { invs_route: IncomingRoute, /// A route for other messages such as Transaction and TransactionNotFound msg_route: IncomingRoute, + + /// Track the number of spam txs coming from this peer + spam_counter: u64, } #[async_trait::async_trait] @@ -59,7 +62,7 @@ impl Flow for RelayTransactionsFlow { impl RelayTransactionsFlow { pub fn new(ctx: FlowContext, router: Arc, invs_route: IncomingRoute, msg_route: IncomingRoute) -> Self { - Self { ctx, router, invs_route, msg_route } + Self { ctx, router, invs_route, msg_route, spam_counter: 0 } } pub fn invs_channel_size() -> usize { @@ -190,9 +193,18 @@ impl RelayTransactionsFlow { self.ctx.broadcast_transactions(accepted_transactions.iter().map(|x| x.id())).await?; } Err(MiningManagerError::MempoolError(err)) => { - if let RuleError::RejectInvalid(_) = err { - // TODO: discuss a banning process - return Err(ProtocolError::MisbehavingPeer(format!("rejected invalid transaction {}", transaction_id))); + match err { + RuleError::RejectInvalid(_) => { + // TODO: discuss a banning process + return Err(ProtocolError::MisbehavingPeer(format!("rejected invalid transaction {}", transaction_id))); + } + RuleError::RejectSpamTransaction(_) => { + self.spam_counter += 1; + if self.spam_counter % 100 == 0 { + kaspa_core::warn!("Peer {} has shared {} spam txs", self.router, self.spam_counter); + } + } + _ => (), } continue; } diff --git a/rpc/grpc/server/Cargo.toml b/rpc/grpc/server/Cargo.toml index 9b46cde36..3d1e39da1 100644 --- a/rpc/grpc/server/Cargo.toml +++ b/rpc/grpc/server/Cargo.toml @@ -16,6 +16,7 @@ kaspa-grpc-core.workspace = true kaspa-utils.workspace = true kaspa-core.workspace = true +uuid.workspace = true faster-hex.workspace = true async-channel.workspace = true parking_lot.workspace = true diff --git a/rpc/grpc/server/src/connection.rs b/rpc/grpc/server/src/connection.rs index 8c7e5fc2d..1aa51505b 100644 --- a/rpc/grpc/server/src/connection.rs +++ b/rpc/grpc/server/src/connection.rs @@ -26,12 +26,16 @@ use tokio::sync::{ oneshot::{channel as oneshot_channel, Sender as OneshotSender}, }; use tonic::Streaming; +use uuid::Uuid; pub type GrpcSender = MpscSender>; pub type StatusResult = Result; #[derive(Debug)] struct Inner { + /// The internal id of this client + id: Uuid, + /// The socket address of this client net_address: SocketAddr, @@ -67,7 +71,13 @@ impl Connection { ) -> Self { let (shutdown_sender, mut shutdown_receiver) = oneshot_channel(); let connection = Self { - inner: Arc::new(Inner { net_address, outgoing_route, manager, shutdown_signal: Mutex::new(Some(shutdown_sender)) }), + inner: Arc::new(Inner { + id: Uuid::new_v4(), + net_address, + outgoing_route, + manager, + shutdown_signal: Mutex::new(Some(shutdown_sender)), + }), }; let connection_clone = connection.clone(); let outgoing_route = connection.inner.outgoing_route.clone(); @@ -150,8 +160,8 @@ impl Connection { self.inner.net_address } - pub fn identity(&self) -> SocketAddr { - self.inner.net_address + pub fn identity(&self) -> Uuid { + self.inner.id } async fn handle_request(request: KaspadRequest, core_service: &DynRpcService) -> GrpcServerResult { diff --git a/rpc/grpc/server/src/manager.rs b/rpc/grpc/server/src/manager.rs index 1e92718fe..79ae2d556 100644 --- a/rpc/grpc/server/src/manager.rs +++ b/rpc/grpc/server/src/manager.rs @@ -7,10 +7,11 @@ use std::{ net::SocketAddr, sync::Arc, }; +use uuid::Uuid; #[derive(Clone, Debug)] pub struct Manager { - connections: Arc>>, + connections: Arc>>, max_connections: usize, } @@ -66,9 +67,4 @@ impl Manager { pub fn has_connections(&self) -> bool { !self.connections.read().is_empty() } - - /// Returns whether a connection matching `net_address` is registered - pub fn has_connection(&self, net_address: SocketAddr) -> bool { - self.connections.read().contains_key(&net_address) - } }