From 3a83759f92c0e41bd1652b8f1fe444fab8c5cc18 Mon Sep 17 00:00:00 2001 From: teor Date: Fri, 25 Nov 2022 15:23:15 +1000 Subject: [PATCH] Implement ZIP-317 transaction selection for block production --- Cargo.lock | 1 + zebra-rpc/Cargo.toml | 5 + .../methods/get_block_template_rpcs/zip317.rs | 117 +++++++++++++++++- .../mempool/storage/verified_set.rs | 2 +- 4 files changed, 121 insertions(+), 4 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 991ee2d2b79..0757d49ed11 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -5458,6 +5458,7 @@ dependencies = [ "num_cpus", "proptest", "proptest-derive", + "rand 0.8.5", "serde", "serde_json", "thiserror", diff --git a/zebra-rpc/Cargo.toml b/zebra-rpc/Cargo.toml index f4eff81e93a..1c6a55fbcc1 100644 --- a/zebra-rpc/Cargo.toml +++ b/zebra-rpc/Cargo.toml @@ -14,6 +14,7 @@ default = [] # Experimental mining RPC support getblocktemplate-rpcs = [ + "rand", "zebra-consensus/getblocktemplate-rpcs", "zebra-state/getblocktemplate-rpcs", "zebra-node-services/getblocktemplate-rpcs", @@ -54,6 +55,10 @@ tracing-futures = "0.2.5" hex = { version = "0.4.3", features = ["serde"] } serde = { version = "1.0.147", features = ["serde_derive"] } +# Experimental feature getblocktemplate-rpcs +rand = { version = "0.8.5", package = "rand", optional = true } + +# Test-only feature proptest-impl proptest = { version = "0.10.1", optional = true } proptest-derive = { version = "0.3.0", optional = true } diff --git a/zebra-rpc/src/methods/get_block_template_rpcs/zip317.rs b/zebra-rpc/src/methods/get_block_template_rpcs/zip317.rs index 5045d89e749..3e9902f4031 100644 --- a/zebra-rpc/src/methods/get_block_template_rpcs/zip317.rs +++ b/zebra-rpc/src/methods/get_block_template_rpcs/zip317.rs @@ -7,9 +7,14 @@ //! > exactly the same between implementations. use jsonrpc_core::{Error, ErrorCode, Result}; +use rand::{ + distributions::{Distribution, WeightedIndex}, + prelude::thread_rng, +}; use tower::{Service, ServiceExt}; -use zebra_chain::transaction::VerifiedUnminedTx; +use zebra_chain::{block::MAX_BLOCK_BYTES, transaction::VerifiedUnminedTx}; +use zebra_consensus::MAX_BLOCK_SIGOPS; use zebra_node_services::mempool; /// Selects mempool transactions for block production according to [ZIP-317]. @@ -30,9 +35,115 @@ where { let mempool_transactions = fetch_mempool_transactions(mempool).await?; - // TODO: ZIP-317 + // Setup the transaction lists. + let (conventional_fee_txs, low_fee_txs): (Vec<_>, Vec<_>) = mempool_transactions + .into_iter() + .partition(VerifiedUnminedTx::pays_conventional_fee); - Ok(mempool_transactions) + // Set up limit tracking + let mut selected_txs = Vec::new(); + let mut remaining_block_sigops = MAX_BLOCK_SIGOPS; + let mut remaining_block_bytes: usize = MAX_BLOCK_BYTES.try_into().expect("fits in memory"); + + // TODO: split these into separate functions? + + if !conventional_fee_txs.is_empty() { + // Setup the transaction weights. + let conventional_fee_tx_weights: Vec = conventional_fee_txs + .iter() + .map(|tx| tx.block_production_fee_weight) + .collect(); + let mut conventional_fee_tx_weights = WeightedIndex::new(conventional_fee_tx_weights) + .expect( + "there is at least one weight, all weights are non-negative, and the total is positive", + ); + + // > Repeat while there is any mempool transaction that: + // > - pays at least the conventional fee, + // > - is within the block sigop limit, and + // > - fits in the block... + loop { + // > Pick one of those transactions at random with probability in direct proportion + // > to its weight, and add it to the block. + let candidate_index = conventional_fee_tx_weights.sample(&mut thread_rng()); + let candidate_tx = &conventional_fee_txs[candidate_index]; + + if candidate_tx.legacy_sigop_count <= remaining_block_sigops + && candidate_tx.transaction.size <= remaining_block_bytes + { + selected_txs.push(candidate_tx.clone()); + + remaining_block_sigops -= candidate_tx.legacy_sigop_count; + remaining_block_bytes -= candidate_tx.transaction.size; + } + + // Only pick each transaction once, by setting picked transaction weights to zero + if conventional_fee_tx_weights + .update_weights(&[(candidate_index, &0.0)]) + .is_err() + { + // All weights are zero, so each transaction has either been selected or rejected + break; + } + } + } + + if !low_fee_txs.is_empty() { + // > Let `N` be the number of remaining transactions with `tx.weight < 1`. + // > Calculate their sum of weights. + let low_fee_tx_weights: Vec = low_fee_txs + .iter() + .map(|tx| tx.block_production_fee_weight) + .collect(); + let low_fee_tx_count = low_fee_tx_weights.len() as f32; + let remaining_weight: f32 = low_fee_tx_weights.iter().sum(); + + // > Calculate `size_target = ...` + // + // We track the remaining bytes within our scaled quota, + // so there is no need to actually calculate `size_target` or `size_of_block_so_far`. + let average_remaining_weight = remaining_weight / low_fee_tx_count; + + let remaining_block_bytes = + remaining_block_bytes as f32 * average_remaining_weight.min(1.0); + let mut remaining_block_bytes = remaining_block_bytes as usize; + + // Setup the transaction weights. + let mut low_fee_tx_weights = WeightedIndex::new(low_fee_tx_weights).expect( + "there is at least one weight, all weights are non-negative, and the total is positive", + ); + + loop { + // > Pick a transaction with probability in direct proportion to its weight + // > and add it to the block. If that transaction would exceed the `size_target` + // > or the block sigop limit, stop without adding it. + let candidate_index = low_fee_tx_weights.sample(&mut thread_rng()); + let candidate_tx = &low_fee_txs[candidate_index]; + + if candidate_tx.legacy_sigop_count > remaining_block_sigops + || candidate_tx.transaction.size > remaining_block_bytes + { + // We've exceeded the (scaled quota) limits + break; + } + + selected_txs.push(candidate_tx.clone()); + + remaining_block_sigops -= candidate_tx.legacy_sigop_count; + remaining_block_bytes -= candidate_tx.transaction.size; + + // Only pick each transaction once, by setting picked transaction weights to zero + if low_fee_tx_weights + .update_weights(&[(candidate_index, &0.0)]) + .is_err() + { + // All weights are zero, so every remaining transaction has been selected + break; + } + } + } + + Ok(selected_txs) } async fn fetch_mempool_transactions(mempool: Mempool) -> Result> diff --git a/zebrad/src/components/mempool/storage/verified_set.rs b/zebrad/src/components/mempool/storage/verified_set.rs index 548fc43fb89..c3abb31ba39 100644 --- a/zebrad/src/components/mempool/storage/verified_set.rs +++ b/zebrad/src/components/mempool/storage/verified_set.rs @@ -161,7 +161,7 @@ impl VerifiedSet { .collect(); let dist = WeightedIndex::new(weights) - .expect("there is at least one weight and all weights are valid"); + .expect("there is at least one weight, all weights are non-negative, and the total is positive"); Some(self.remove(dist.sample(&mut thread_rng()))) }