From 6de3f428b9600d00b5f75b24617000ee1df7b996 Mon Sep 17 00:00:00 2001 From: Yash Atreya <44857776+yash-atreya@users.noreply.github.com> Date: Fri, 5 Apr 2024 22:49:29 -0400 Subject: [PATCH 01/12] num-audit: extension * gas related field to `u128` * `nonce` and `number` to `u64` in `Header` from `B64` and `U256` respectively --- crates/consensus/src/header.rs | 24 ++++----- crates/consensus/src/receipt/any.rs | 2 +- crates/consensus/src/receipt/envelope.rs | 2 +- crates/consensus/src/receipt/mod.rs | 6 +-- crates/consensus/src/receipt/receipts.rs | 8 +-- crates/eips/src/eip1559/basefee.rs | 8 +-- crates/eips/src/eip1559/helpers.rs | 19 ++++--- crates/eips/src/eip4844/mod.rs | 31 +++++------ crates/provider/src/heart.rs | 8 +-- crates/provider/src/provider.rs | 11 ++-- crates/rpc-types/src/eth/block.rs | 52 +++++++++---------- .../rpc-types/src/eth/transaction/receipt.rs | 2 +- 12 files changed, 84 insertions(+), 89 deletions(-) diff --git a/crates/consensus/src/header.rs b/crates/consensus/src/header.rs index 0d5bb832769..44276b2eb59 100644 --- a/crates/consensus/src/header.rs +++ b/crates/consensus/src/header.rs @@ -54,9 +54,9 @@ pub struct Header { /// zero; formally Hi. pub number: BlockNumber, /// A scalar value equal to the current limit of gas expenditure per block; formally Hl. - pub gas_limit: u64, + pub gas_limit: u128, /// A scalar value equal to the total gas used in transactions in this block; formally Hg. - pub gas_used: u64, + pub gas_used: u128, /// A scalar value equal to the reasonable output of Unix’s time() at this block’s inception; /// formally Hs. pub timestamp: u64, @@ -73,14 +73,14 @@ pub struct Header { /// The algorithm results in the base fee per gas increasing when blocks are /// above the gas target, and decreasing when blocks are below the gas target. The base fee per /// gas is burned. - pub base_fee_per_gas: Option, + pub base_fee_per_gas: Option, /// The total amount of blob gas consumed by the transactions within the block, added in /// EIP-4844. - pub blob_gas_used: Option, + pub blob_gas_used: Option, /// A running total of blob gas consumed in excess of the target, prior to the block. Blocks /// with above-target blob gas consumption increase this value, blocks with below-target blob /// gas consumption decrease it (bounded at 0). This was added in EIP-4844. - pub excess_blob_gas: Option, + pub excess_blob_gas: Option, /// The hash of the parent beacon block's root is included in execution blocks, as proposed by /// EIP-4788. /// @@ -194,7 +194,7 @@ impl Header { /// Calculate base fee for next block according to the EIP-1559 spec. /// /// Returns a `None` if no base fee is set, no EIP-1559 support - pub fn next_block_base_fee(&self, base_fee_params: BaseFeeParams) -> Option { + pub fn next_block_base_fee(&self, base_fee_params: BaseFeeParams) -> Option { Some(calc_next_block_base_fee( self.gas_used, self.gas_limit, @@ -207,7 +207,7 @@ impl Header { /// spec. /// /// Returns a `None` if no excess blob gas is set, no EIP-4844 support - pub fn next_block_excess_blob_gas(&self) -> Option { + pub fn next_block_excess_blob_gas(&self) -> Option { Some(calc_excess_blob_gas(self.excess_blob_gas?, self.blob_gas_used?)) } @@ -397,8 +397,8 @@ impl Decodable for Header { logs_bloom: Decodable::decode(buf)?, difficulty: Decodable::decode(buf)?, number: U256::decode(buf)?.to::(), - gas_limit: U256::decode(buf)?.to::(), - gas_used: U256::decode(buf)?.to::(), + gas_limit: U256::decode(buf)?.to::(), + gas_used: U256::decode(buf)?.to::(), timestamp: Decodable::decode(buf)?, extra_data: Decodable::decode(buf)?, mix_hash: Decodable::decode(buf)?, @@ -414,7 +414,7 @@ impl Decodable for Header { if buf.first().map(|b| *b == EMPTY_LIST_CODE).unwrap_or_default() { buf.advance(1) } else { - this.base_fee_per_gas = Some(U256::decode(buf)?.to::()); + this.base_fee_per_gas = Some(U256::decode(buf)?.to::()); } } @@ -432,7 +432,7 @@ impl Decodable for Header { if buf.first().map(|b| *b == EMPTY_LIST_CODE).unwrap_or_default() { buf.advance(1) } else { - this.blob_gas_used = Some(U256::decode(buf)?.to::()); + this.blob_gas_used = Some(U256::decode(buf)?.to::()); } } @@ -440,7 +440,7 @@ impl Decodable for Header { if buf.first().map(|b| *b == EMPTY_LIST_CODE).unwrap_or_default() { buf.advance(1) } else { - this.excess_blob_gas = Some(U256::decode(buf)?.to::()); + this.excess_blob_gas = Some(U256::decode(buf)?.to::()); } } diff --git a/crates/consensus/src/receipt/any.rs b/crates/consensus/src/receipt/any.rs index 2fae44a51e1..35482caab59 100644 --- a/crates/consensus/src/receipt/any.rs +++ b/crates/consensus/src/receipt/any.rs @@ -51,7 +51,7 @@ impl AnyReceiptEnvelope { } /// Returns the cumulative gas used at this receipt. - pub const fn cumulative_gas_used(&self) -> u64 { + pub const fn cumulative_gas_used(&self) -> u128 { self.inner.receipt.cumulative_gas_used } diff --git a/crates/consensus/src/receipt/envelope.rs b/crates/consensus/src/receipt/envelope.rs index 0738a0308e4..6890f5a9433 100644 --- a/crates/consensus/src/receipt/envelope.rs +++ b/crates/consensus/src/receipt/envelope.rs @@ -60,7 +60,7 @@ impl ReceiptEnvelope { } /// Returns the cumulative gas used at this receipt. - pub fn cumulative_gas_used(&self) -> u64 { + pub fn cumulative_gas_used(&self) -> u128 { self.as_receipt().unwrap().cumulative_gas_used } diff --git a/crates/consensus/src/receipt/mod.rs b/crates/consensus/src/receipt/mod.rs index 08acf747524..3eda915b193 100644 --- a/crates/consensus/src/receipt/mod.rs +++ b/crates/consensus/src/receipt/mod.rs @@ -25,7 +25,7 @@ pub trait TxReceipt { } /// Returns the cumulative gas used in the block after this transaction was executed. - fn cumulative_gas_used(&self) -> u64; + fn cumulative_gas_used(&self) -> u128; /// Returns the logs emitted by this transaction. fn logs(&self) -> &[Log]; @@ -47,7 +47,7 @@ mod tests { let receipt = ReceiptEnvelope::Legacy(ReceiptWithBloom { receipt: Receipt { - cumulative_gas_used: 0x1u64, + cumulative_gas_used: 0x1u128, logs: vec![Log { address: address!("0000000000000000000000000000000000000011"), data: LogData::new_unchecked( @@ -79,7 +79,7 @@ mod tests { let expected = ReceiptWithBloom { receipt: Receipt { - cumulative_gas_used: 0x1u64, + cumulative_gas_used: 0x1u128, logs: vec![Log { address: address!("0000000000000000000000000000000000000011"), data: LogData::new_unchecked( diff --git a/crates/consensus/src/receipt/receipts.rs b/crates/consensus/src/receipt/receipts.rs index db9eb3afc22..3acd06e5a5c 100644 --- a/crates/consensus/src/receipt/receipts.rs +++ b/crates/consensus/src/receipt/receipts.rs @@ -17,8 +17,8 @@ pub struct Receipt { #[cfg_attr(feature = "serde", serde(with = "alloy_serde::quantity_bool"))] pub status: bool, /// Gas used - #[cfg_attr(feature = "serde", serde(with = "alloy_serde::u64_hex"))] - pub cumulative_gas_used: u64, + #[cfg_attr(feature = "serde", serde(with = "alloy_serde::u128_hex_or_decimal"))] + pub cumulative_gas_used: u128, /// Log send from contracts. pub logs: Vec, } @@ -46,7 +46,7 @@ impl TxReceipt for Receipt { self.bloom_slow() } - fn cumulative_gas_used(&self) -> u64 { + fn cumulative_gas_used(&self) -> u128 { self.cumulative_gas_used } @@ -85,7 +85,7 @@ impl TxReceipt for ReceiptWithBloom { Some(self.logs_bloom) } - fn cumulative_gas_used(&self) -> u64 { + fn cumulative_gas_used(&self) -> u128 { self.receipt.cumulative_gas_used } diff --git a/crates/eips/src/eip1559/basefee.rs b/crates/eips/src/eip1559/basefee.rs index 02696d1295b..a8f4b64894f 100644 --- a/crates/eips/src/eip1559/basefee.rs +++ b/crates/eips/src/eip1559/basefee.rs @@ -7,17 +7,17 @@ use crate::eip1559::constants::{ #[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))] pub struct BaseFeeParams { /// The base_fee_max_change_denominator from EIP-1559 - pub max_change_denominator: u64, + pub max_change_denominator: u128, /// The elasticity multiplier from EIP-1559 - pub elasticity_multiplier: u64, + pub elasticity_multiplier: u128, } impl BaseFeeParams { /// Get the base fee parameters for Ethereum mainnet pub const fn ethereum() -> BaseFeeParams { BaseFeeParams { - max_change_denominator: DEFAULT_BASE_FEE_MAX_CHANGE_DENOMINATOR, - elasticity_multiplier: DEFAULT_ELASTICITY_MULTIPLIER, + max_change_denominator: DEFAULT_BASE_FEE_MAX_CHANGE_DENOMINATOR as u128, + elasticity_multiplier: DEFAULT_ELASTICITY_MULTIPLIER as u128, } } } diff --git a/crates/eips/src/eip1559/helpers.rs b/crates/eips/src/eip1559/helpers.rs index 98980e2371a..ef7ac11b09e 100644 --- a/crates/eips/src/eip1559/helpers.rs +++ b/crates/eips/src/eip1559/helpers.rs @@ -24,11 +24,11 @@ use crate::eip1559::BaseFeeParams; /// /// For more information, refer to the [EIP-1559 spec](https://github.com/ethereum/EIPs/blob/master/EIPS/eip-1559.md). pub fn calc_next_block_base_fee( - gas_used: u64, - gas_limit: u64, - base_fee: u64, + gas_used: u128, + gas_limit: u128, + base_fee: u128, base_fee_params: BaseFeeParams, -) -> u64 { +) -> u128 { // Calculate the target gas by dividing the gas limit by the elasticity multiplier. let gas_target = gas_limit / base_fee_params.elasticity_multiplier; @@ -44,18 +44,17 @@ pub fn calc_next_block_base_fee( + (core::cmp::max( // Ensure a minimum increase of 1. 1, - base_fee as u128 * (gas_used - gas_target) as u128 - / (gas_target as u128 * base_fee_params.max_change_denominator as u128), - ) as u64) + base_fee * (gas_used - gas_target) + / (gas_target * base_fee_params.max_change_denominator), + )) } // If the gas used in the current block is less than the gas target, calculate a new // decreased base fee. core::cmp::Ordering::Less => { // Calculate the decrease in base fee based on the formula defined by EIP-1559. base_fee.saturating_sub( - (base_fee as u128 * (gas_target - gas_used) as u128 - / (gas_target as u128 * base_fee_params.max_change_denominator as u128)) - as u64, + base_fee * (gas_target - gas_used) + / (gas_target * base_fee_params.max_change_denominator), ) } } diff --git a/crates/eips/src/eip4844/mod.rs b/crates/eips/src/eip4844/mod.rs index 3d2fc2322da..0bd1097ab67 100644 --- a/crates/eips/src/eip4844/mod.rs +++ b/crates/eips/src/eip4844/mod.rs @@ -55,7 +55,7 @@ pub const MAX_BLOBS_PER_BLOCK: usize = (MAX_DATA_GAS_PER_BLOCK / DATA_GAS_PER_BL pub const TARGET_BLOBS_PER_BLOCK: u64 = TARGET_DATA_GAS_PER_BLOCK / DATA_GAS_PER_BLOB; // 393216 / 131072 = 3 /// Determines the maximum rate of change for blob fee -pub const BLOB_GASPRICE_UPDATE_FRACTION: u64 = 3_338_477u64; // 3338477 +pub const BLOB_GASPRICE_UPDATE_FRACTION: u128 = 3_338_477u128; // 3338477 /// Minimum gas price for a data blob pub const BLOB_TX_MIN_BLOB_GASPRICE: u128 = 1u128; @@ -80,8 +80,12 @@ pub type Bytes48 = FixedBytes<48>; /// See also [the EIP-4844 helpers](https://eips.ethereum.org/EIPS/eip-4844#helpers) /// (`calc_excess_blob_gas`). #[inline] -pub const fn calc_excess_blob_gas(parent_excess_blob_gas: u64, parent_blob_gas_used: u64) -> u64 { - (parent_excess_blob_gas + parent_blob_gas_used).saturating_sub(TARGET_DATA_GAS_PER_BLOCK) +pub const fn calc_excess_blob_gas( + parent_excess_blob_gas: u128, + parent_blob_gas_used: u128, +) -> u128 { + (parent_excess_blob_gas + parent_blob_gas_used) + .saturating_sub(TARGET_DATA_GAS_PER_BLOCK as u128) } /// Calculates the blob gas price from the header's excess blob gas field. @@ -89,12 +93,8 @@ pub const fn calc_excess_blob_gas(parent_excess_blob_gas: u64, parent_blob_gas_u /// See also [the EIP-4844 helpers](https://eips.ethereum.org/EIPS/eip-4844#helpers) /// (`get_blob_gasprice`). #[inline] -pub fn calc_blob_gasprice(excess_blob_gas: u64) -> u128 { - fake_exponential( - BLOB_TX_MIN_BLOB_GASPRICE as u64, - excess_blob_gas, - BLOB_GASPRICE_UPDATE_FRACTION, - ) +pub fn calc_blob_gasprice(excess_blob_gas: u128) -> u128 { + fake_exponential(BLOB_TX_MIN_BLOB_GASPRICE, excess_blob_gas, BLOB_GASPRICE_UPDATE_FRACTION) } /// Approximates `factor * e ** (numerator / denominator)` using Taylor expansion. @@ -108,11 +108,8 @@ pub fn calc_blob_gasprice(excess_blob_gas: u64) -> u128 { /// /// This function panics if `denominator` is zero. #[inline] -fn fake_exponential(factor: u64, numerator: u64, denominator: u64) -> u128 { +fn fake_exponential(factor: u128, numerator: u128, denominator: u128) -> u128 { assert_ne!(denominator, 0, "attempt to divide by zero"); - let factor = factor as u128; - let numerator = numerator as u128; - let denominator = denominator as u128; let mut i = 1; let mut output = 0; @@ -164,8 +161,8 @@ mod tests { ), (DATA_GAS_PER_BLOB - 1, (TARGET_DATA_GAS_PER_BLOCK / DATA_GAS_PER_BLOB) - 1, 0), ] { - let actual = calc_excess_blob_gas(excess, blobs * DATA_GAS_PER_BLOB); - assert_eq!(actual, expected, "test: {t:?}"); + let actual = calc_excess_blob_gas(excess as u128, (blobs * DATA_GAS_PER_BLOB) as u128); + assert_eq!(actual, expected as u128, "test: {t:?}"); } } @@ -213,9 +210,9 @@ mod tests { (1, 5, 2, 11), // approximate 12.18 (2, 5, 2, 23), // approximate 24.36 (1, 50000000, 2225652, 5709098764), - (1, 380928, BLOB_GASPRICE_UPDATE_FRACTION, 1), + (1, 380928, BLOB_GASPRICE_UPDATE_FRACTION.try_into().unwrap(), 1), ] { - let actual = fake_exponential(factor, numerator, denominator); + let actual = fake_exponential(factor as u128, numerator as u128, denominator as u128); assert_eq!(actual, expected, "test: {t:?}"); } } diff --git a/crates/provider/src/heart.rs b/crates/provider/src/heart.rs index f49744b72b1..bf3c860a896 100644 --- a/crates/provider/src/heart.rs +++ b/crates/provider/src/heart.rs @@ -383,8 +383,8 @@ impl + Unpin + 'static> Heartbeat { impl Heartbeat { /// Check if any transactions have enough confirmations to notify. - fn check_confirmations(&mut self, current_height: &U256) { - let to_keep = self.waiting_confs.split_off(&(current_height + U256::from(1))); + fn check_confirmations(&mut self, current_height: u64) { + let to_keep = self.waiting_confs.split_off(&U256::from(current_height + 1)); let to_notify = std::mem::replace(&mut self.waiting_confs, to_keep); for watcher in to_notify.into_values().flatten() { watcher.notify(); @@ -445,12 +445,12 @@ impl Heartbeat { // Otherwise add it to the waiting list. debug!(tx=%watcher.config.tx_hash, %block_height, confirmations, "adding to waiting list"); self.waiting_confs - .entry(*block_height + U256::from(confirmations) - U256::from(1)) + .entry(U256::from(*block_height + confirmations - 1)) .or_default() .push(watcher); } - self.check_confirmations(block_height); + self.check_confirmations(*block_height); // Update the latest block. We use `send_replace` here to ensure the // latest block is always up to date, even if no receivers exist. diff --git a/crates/provider/src/provider.rs b/crates/provider/src/provider.rs index 66d4714f767..fec5a15066d 100644 --- a/crates/provider/src/provider.rs +++ b/crates/provider/src/provider.rs @@ -866,7 +866,6 @@ pub trait Provider: .ok_or(RpcError::NullResp)? .header .base_fee_per_gas - .map(|fee| fee.to::()) .ok_or(RpcError::UnsupportedFeature("eip1559"))? } }; @@ -1097,7 +1096,7 @@ mod tests { let mut stream = sub.into_stream().take(2); let mut n = 1; while let Some(block) = stream.next().await { - assert_eq!(block.header.number.unwrap(), U256::from(n)); + assert_eq!(block.header.number.unwrap(), n); assert_eq!(block.transactions.hashes().len(), 0); n += 1; } @@ -1119,7 +1118,7 @@ mod tests { let mut stream = sub.into_stream().take(2); let mut n = 1; while let Some(block) = stream.next().await { - assert_eq!(block.header.number.unwrap(), U256::from(n)); + assert_eq!(block.header.number.unwrap(), n); assert_eq!(block.transactions.hashes().len(), 0); n += 1; } @@ -1139,7 +1138,7 @@ mod tests { let mut stream = sub.into_stream().take(1); while let Some(block) = stream.next().await { println!("New block {:?}", block); - assert!(block.header.number.unwrap() > U256::ZERO); + assert!(block.header.number.unwrap() > 0); } } @@ -1238,7 +1237,7 @@ mod tests { let num = 0; let tag: BlockNumberOrTag = num.into(); let block = provider.get_block_by_number(tag, true).await.unwrap().unwrap(); - assert_eq!(block.header.number.unwrap(), U256::from(num)); + assert_eq!(block.header.number.unwrap(), num); } #[tokio::test] @@ -1249,7 +1248,7 @@ mod tests { let num = 0; let tag: BlockNumberOrTag = num.into(); let block = provider.get_block_by_number(tag, true).await.unwrap().unwrap(); - assert_eq!(block.header.number.unwrap(), U256::from(num)); + assert_eq!(block.header.number.unwrap(), num); } #[tokio::test] diff --git a/crates/rpc-types/src/eth/block.rs b/crates/rpc-types/src/eth/block.rs index 084dcb7ab45..d188367154f 100644 --- a/crates/rpc-types/src/eth/block.rs +++ b/crates/rpc-types/src/eth/block.rs @@ -5,7 +5,7 @@ use crate::{other::OtherFields, Transaction, Withdrawal}; use alloy_eips::{calc_blob_gasprice, calc_excess_blob_gas}; use alloy_primitives::{ - ruint::ParseError, Address, BlockHash, BlockNumber, Bloom, Bytes, B256, B64, U256, U64, + ruint::ParseError, Address, BlockHash, BlockNumber, Bloom, Bytes, B256, U256, U64, }; use alloy_rlp::{bytes, Decodable, Encodable, Error as RlpError}; use serde::{ @@ -75,11 +75,11 @@ pub struct Header { /// Difficulty pub difficulty: U256, /// Block number - pub number: Option, + pub number: Option, /// Gas Limit - pub gas_limit: U256, + pub gas_limit: u128, /// Gas Used - pub gas_used: U256, + pub gas_used: u128, /// Timestamp pub timestamp: U256, /// Total difficulty @@ -101,19 +101,19 @@ pub struct Header { #[serde(skip_serializing_if = "Option::is_none")] pub mix_hash: Option, /// Nonce - pub nonce: Option, + pub nonce: Option, /// Base fee per unit of gas (if past London) #[serde(skip_serializing_if = "Option::is_none")] - pub base_fee_per_gas: Option, + pub base_fee_per_gas: Option, /// Withdrawals root hash added by EIP-4895 and is ignored in legacy headers. #[serde(skip_serializing_if = "Option::is_none")] pub withdrawals_root: Option, /// Blob gas used #[serde(skip_serializing_if = "Option::is_none")] - pub blob_gas_used: Option, + pub blob_gas_used: Option, /// Excess blob gas #[serde(skip_serializing_if = "Option::is_none")] - pub excess_blob_gas: Option, + pub excess_blob_gas: Option, /// Parent beacon block root #[serde(skip_serializing_if = "Option::is_none")] pub parent_beacon_block_root: Option, @@ -124,7 +124,7 @@ impl Header { /// /// Returns `None` if `excess_blob_gas` is None pub fn blob_fee(&self) -> Option { - self.excess_blob_gas.map(|gas| calc_blob_gasprice(gas.to())) + self.excess_blob_gas.map(calc_blob_gasprice) } /// Returns the blob fee for the next block according to the EIP-4844 spec. @@ -140,8 +140,8 @@ impl Header { /// spec. /// /// Returns a `None` if no excess blob gas is set, no EIP-4844 support - pub fn next_block_excess_blob_gas(&self) -> Option { - Some(calc_excess_blob_gas(self.excess_blob_gas?.to(), self.blob_gas_used?.to())) + pub fn next_block_excess_blob_gas(&self) -> Option { + Some(calc_excess_blob_gas(self.excess_blob_gas?, self.blob_gas_used?)) } } @@ -1172,17 +1172,17 @@ mod tests { transactions_root: B256::with_last_byte(6), receipts_root: B256::with_last_byte(7), withdrawals_root: Some(B256::with_last_byte(8)), - number: Some(U256::from(9)), - gas_used: U256::from(10), - gas_limit: U256::from(11), + number: Some(9), + gas_used: 10, + gas_limit: 11, extra_data: Bytes::from(vec![1, 2, 3]), logs_bloom: Bloom::default(), timestamp: U256::from(12), difficulty: U256::from(13), total_difficulty: Some(U256::from(100000)), mix_hash: Some(B256::with_last_byte(14)), - nonce: Some(B64::with_last_byte(15)), - base_fee_per_gas: Some(U256::from(20)), + nonce: Some(15), + base_fee_per_gas: Some(20), blob_gas_used: None, excess_blob_gas: None, parent_beacon_block_root: None, @@ -1214,17 +1214,17 @@ mod tests { transactions_root: B256::with_last_byte(6), receipts_root: B256::with_last_byte(7), withdrawals_root: Some(B256::with_last_byte(8)), - number: Some(U256::from(9)), - gas_used: U256::from(10), - gas_limit: U256::from(11), + number: Some(9), + gas_used: 10, + gas_limit: 11, extra_data: Bytes::from(vec![1, 2, 3]), logs_bloom: Bloom::default(), timestamp: U256::from(12), difficulty: U256::from(13), total_difficulty: Some(U256::from(100000)), mix_hash: Some(B256::with_last_byte(14)), - nonce: Some(B64::with_last_byte(15)), - base_fee_per_gas: Some(U256::from(20)), + nonce: Some(15), + base_fee_per_gas: Some(20), blob_gas_used: None, excess_blob_gas: None, parent_beacon_block_root: None, @@ -1256,17 +1256,17 @@ mod tests { transactions_root: B256::with_last_byte(6), receipts_root: B256::with_last_byte(7), withdrawals_root: None, - number: Some(U256::from(9)), - gas_used: U256::from(10), - gas_limit: U256::from(11), + number: Some(9), + gas_used: 10, + gas_limit: 11, extra_data: Bytes::from(vec![1, 2, 3]), logs_bloom: Bloom::default(), timestamp: U256::from(12), difficulty: U256::from(13), total_difficulty: Some(U256::from(100000)), mix_hash: Some(B256::with_last_byte(14)), - nonce: Some(B64::with_last_byte(15)), - base_fee_per_gas: Some(U256::from(20)), + nonce: Some(15), + base_fee_per_gas: Some(20), blob_gas_used: None, excess_blob_gas: None, parent_beacon_block_root: None, diff --git a/crates/rpc-types/src/eth/transaction/receipt.rs b/crates/rpc-types/src/eth/transaction/receipt.rs index 1ed7effd97f..e2da0e3ee5a 100644 --- a/crates/rpc-types/src/eth/transaction/receipt.rs +++ b/crates/rpc-types/src/eth/transaction/receipt.rs @@ -143,7 +143,7 @@ mod test { ); const EXPECTED_BLOOM: Bloom = bloom!("00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000010000000000000000000000000000200000000000000000040000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000400000800000000000000000000000000000000004000000000000000000800000000100000020000000000000000000080000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000010000000000000000000000000000000010000000000000000000000000000"); - const EXPECTED_CGU: u64 = 0xa42aec; + const EXPECTED_CGU: u128 = 0xa42aec; assert!(matches!( receipt.inner, From 645491786a4daa7bd3724d06d94b840292ac288a Mon Sep 17 00:00:00 2001 From: Yash Atreya <44857776+yash-atreya@users.noreply.github.com> Date: Fri, 5 Apr 2024 23:33:54 -0400 Subject: [PATCH 02/12] num-audit: gas fields to `u128` in Genesis --- crates/genesis/src/lib.rs | 28 +++++++++++++++------------- 1 file changed, 15 insertions(+), 13 deletions(-) diff --git a/crates/genesis/src/lib.rs b/crates/genesis/src/lib.rs index 5c448bc2fbd..2091478f274 100644 --- a/crates/genesis/src/lib.rs +++ b/crates/genesis/src/lib.rs @@ -24,7 +24,9 @@ use alloc::collections::BTreeMap; use alloy_primitives::{Address, Bytes, B256, U256}; use alloy_serde::{ - num::{u64_hex_or_decimal, u64_hex_or_decimal_opt}, + num::{ + u128_hex_or_decimal, u128_hex_or_decimal_opt, u64_hex_or_decimal, u64_hex_or_decimal_opt, + }, storage::deserialize_storage_map, ttd::deserialize_json_ttd_opt, }; @@ -46,8 +48,8 @@ pub struct Genesis { /// The genesis header extra data. pub extra_data: Bytes, /// The genesis header gas limit. - #[serde(with = "u64_hex_or_decimal")] - pub gas_limit: u64, + #[serde(with = "u128_hex_or_decimal")] + pub gas_limit: u128, /// The genesis header difficulty. pub difficulty: U256, /// The genesis header mix hash. @@ -64,14 +66,14 @@ pub struct Genesis { // should NOT be set in a real genesis file, but are included here for compatibility with // consensus tests, which have genesis files with these fields populated. /// The genesis header base fee - #[serde(skip_serializing_if = "Option::is_none", with = "u64_hex_or_decimal_opt")] - pub base_fee_per_gas: Option, + #[serde(skip_serializing_if = "Option::is_none", with = "u128_hex_or_decimal_opt")] + pub base_fee_per_gas: Option, /// The genesis header excess blob gas - #[serde(skip_serializing_if = "Option::is_none", with = "u64_hex_or_decimal_opt")] - pub excess_blob_gas: Option, + #[serde(skip_serializing_if = "Option::is_none", with = "u128_hex_or_decimal_opt")] + pub excess_blob_gas: Option, /// The genesis header blob gas used - #[serde(skip_serializing_if = "Option::is_none", with = "u64_hex_or_decimal_opt")] - pub blob_gas_used: Option, + #[serde(skip_serializing_if = "Option::is_none", with = "u128_hex_or_decimal_opt")] + pub blob_gas_used: Option, /// The genesis block number #[serde(skip_serializing_if = "Option::is_none", with = "u64_hex_or_decimal_opt")] pub number: Option, @@ -158,7 +160,7 @@ impl Genesis { } /// Set the gas limit. - pub const fn with_gas_limit(mut self, gas_limit: u64) -> Self { + pub const fn with_gas_limit(mut self, gas_limit: u128) -> Self { self.gas_limit = gas_limit; self } @@ -182,19 +184,19 @@ impl Genesis { } /// Set the base fee. - pub const fn with_base_fee(mut self, base_fee: Option) -> Self { + pub const fn with_base_fee(mut self, base_fee: Option) -> Self { self.base_fee_per_gas = base_fee; self } /// Set the excess blob gas. - pub const fn with_excess_blob_gas(mut self, excess_blob_gas: Option) -> Self { + pub const fn with_excess_blob_gas(mut self, excess_blob_gas: Option) -> Self { self.excess_blob_gas = excess_blob_gas; self } /// Set the blob gas used. - pub const fn with_blob_gas_used(mut self, blob_gas_used: Option) -> Self { + pub const fn with_blob_gas_used(mut self, blob_gas_used: Option) -> Self { self.blob_gas_used = blob_gas_used; self } From e58e137083e319e8ac8c899be4feb13b0dd8e271 Mon Sep 17 00:00:00 2001 From: Yash Atreya <44857776+yash-atreya@users.noreply.github.com> Date: Fri, 5 Apr 2024 23:48:18 -0400 Subject: [PATCH 03/12] num-audit: use primitive types in rpc-types: common, optimism, receipt, request --- crates/provider/src/heart.rs | 11 ++++----- .../rpc-types/src/eth/transaction/common.rs | 2 +- .../rpc-types/src/eth/transaction/optimism.rs | 23 ++++++++++++------ .../rpc-types/src/eth/transaction/receipt.rs | 24 ++++++++++++------- .../rpc-types/src/eth/transaction/request.rs | 12 ++++++---- 5 files changed, 45 insertions(+), 27 deletions(-) diff --git a/crates/provider/src/heart.rs b/crates/provider/src/heart.rs index bf3c860a896..0a4591973a4 100644 --- a/crates/provider/src/heart.rs +++ b/crates/provider/src/heart.rs @@ -3,7 +3,7 @@ use crate::{Provider, RootProvider}; use alloy_json_rpc::RpcError; use alloy_network::Network; -use alloy_primitives::{B256, U256}; +use alloy_primitives::B256; use alloy_rpc_types::Block; use alloy_transport::{utils::Spawnable, Transport, TransportErrorKind, TransportResult}; use futures::{stream::StreamExt, FutureExt, Stream}; @@ -363,7 +363,7 @@ pub(crate) struct Heartbeat { unconfirmed: HashMap, /// Ordered map of transactions waiting for confirmations. - waiting_confs: BTreeMap>, + waiting_confs: BTreeMap>, /// Ordered map of transactions to reap at a certain time. reap_at: BTreeMap, @@ -384,7 +384,7 @@ impl + Unpin + 'static> Heartbeat { impl Heartbeat { /// Check if any transactions have enough confirmations to notify. fn check_confirmations(&mut self, current_height: u64) { - let to_keep = self.waiting_confs.split_off(&U256::from(current_height + 1)); + let to_keep = self.waiting_confs.split_off(&(current_height + 1)); let to_notify = std::mem::replace(&mut self.waiting_confs, to_keep); for watcher in to_notify.into_values().flatten() { watcher.notify(); @@ -444,10 +444,7 @@ impl Heartbeat { } // Otherwise add it to the waiting list. debug!(tx=%watcher.config.tx_hash, %block_height, confirmations, "adding to waiting list"); - self.waiting_confs - .entry(U256::from(*block_height + confirmations - 1)) - .or_default() - .push(watcher); + self.waiting_confs.entry(*block_height + confirmations - 1).or_default().push(watcher); } self.check_confirmations(*block_height); diff --git a/crates/rpc-types/src/eth/transaction/common.rs b/crates/rpc-types/src/eth/transaction/common.rs index b6217bdb82f..dd00fee8d74 100644 --- a/crates/rpc-types/src/eth/transaction/common.rs +++ b/crates/rpc-types/src/eth/transaction/common.rs @@ -15,5 +15,5 @@ pub struct TransactionInfo { /// Number of the block. pub block_number: Option, /// Base fee of the block. - pub base_fee: Option, + pub base_fee: Option, } diff --git a/crates/rpc-types/src/eth/transaction/optimism.rs b/crates/rpc-types/src/eth/transaction/optimism.rs index cbc8b6594ad..5cfcf8140c0 100644 --- a/crates/rpc-types/src/eth/transaction/optimism.rs +++ b/crates/rpc-types/src/eth/transaction/optimism.rs @@ -1,7 +1,7 @@ //! Misc Optimism-specific types. use crate::other::OtherFields; -use alloy_primitives::{B256, U128, U256, U64}; +use alloy_primitives::{B256, U128, U64}; use serde::{Deserialize, Serialize}; /// Optimism specific transaction fields @@ -30,17 +30,26 @@ pub struct OptimismTransactionReceiptFields { #[serde(skip_serializing_if = "Option::is_none")] pub deposit_receipt_version: Option, /// L1 fee for the transaction - #[serde(skip_serializing_if = "Option::is_none")] - pub l1_fee: Option, + #[serde( + skip_serializing_if = "Option::is_none", + with = "alloy_serde::num::u128_hex_or_decimal_opt" + )] + pub l1_fee: Option, /// L1 fee scalar for the transaction #[serde(default, skip_serializing_if = "Option::is_none", with = "l1_fee_scalar_serde")] pub l1_fee_scalar: Option, /// L1 gas price for the transaction - #[serde(skip_serializing_if = "Option::is_none")] - pub l1_gas_price: Option, + #[serde( + skip_serializing_if = "Option::is_none", + with = "alloy_serde::num::u128_hex_or_decimal_opt" + )] + pub l1_gas_price: Option, /// L1 gas used for the transaction - #[serde(skip_serializing_if = "Option::is_none")] - pub l1_gas_used: Option, + #[serde( + skip_serializing_if = "Option::is_none", + with = "alloy_serde::num::u128_hex_or_decimal_opt" + )] + pub l1_gas_used: Option, } impl From for OtherFields { diff --git a/crates/rpc-types/src/eth/transaction/receipt.rs b/crates/rpc-types/src/eth/transaction/receipt.rs index e2da0e3ee5a..7c79dd18754 100644 --- a/crates/rpc-types/src/eth/transaction/receipt.rs +++ b/crates/rpc-types/src/eth/transaction/receipt.rs @@ -28,21 +28,29 @@ pub struct TransactionReceipt> { #[serde(with = "alloy_serde::u64_hex_opt")] pub block_number: Option, /// Gas used by this transaction alone. - #[serde(with = "alloy_serde::u64_hex_opt")] - pub gas_used: Option, + #[serde(with = "alloy_serde::u128_hex_or_decimal_opt")] + pub gas_used: Option, /// The price paid post-execution by the transaction (i.e. base fee + priority fee). Both /// fields in 1559-style transactions are maximums (max fee + max priority fee), the amount /// that's actually paid by users can only be determined post-execution - #[serde(with = "alloy_serde::u64_hex")] - pub effective_gas_price: u64, + #[serde(with = "alloy_serde::u128_hex_or_decimal")] + pub effective_gas_price: u128, /// Blob gas used by the eip-4844 transaction /// /// This is None for non eip-4844 transactions - #[serde(skip_serializing_if = "Option::is_none", with = "alloy_serde::u64_hex_opt", default)] - pub blob_gas_used: Option, + #[serde( + skip_serializing_if = "Option::is_none", + with = "alloy_serde::u128_hex_or_decimal_opt", + default + )] + pub blob_gas_used: Option, /// The price paid by the eip-4844 transaction per blob gas. - #[serde(skip_serializing_if = "Option::is_none", with = "alloy_serde::u64_hex_opt", default)] - pub blob_gas_price: Option, + #[serde( + skip_serializing_if = "Option::is_none", + with = "alloy_serde::u128_hex_or_decimal_opt", + default + )] + pub blob_gas_price: Option, /// Address of the sender pub from: Address, /// Address of the receiver. None when its a contract creation transaction. diff --git a/crates/rpc-types/src/eth/transaction/request.rs b/crates/rpc-types/src/eth/transaction/request.rs index 8d3a568407f..fe6a264743a 100644 --- a/crates/rpc-types/src/eth/transaction/request.rs +++ b/crates/rpc-types/src/eth/transaction/request.rs @@ -14,18 +14,22 @@ pub struct TransactionRequest { /// The destination address of the transaction. pub to: Option
, /// The legacy gas price. - #[serde(default)] + #[serde(default, with = "alloy_serde::num::u128_hex_or_decimal_opt")] pub gas_price: Option, /// The max base fee per gas the sender is willing to pay. - #[serde(default)] + #[serde(default, with = "alloy_serde::num::u128_hex_or_decimal_opt")] pub max_fee_per_gas: Option, /// The max priority fee per gas the sender is willing to pay, also called the miner tip. - #[serde(default)] + #[serde(default, with = "alloy_serde::num::u128_hex_or_decimal_opt")] pub max_priority_fee_per_gas: Option, /// The max fee per blob gas for EIP-4844 blob transactions. - #[serde(skip_serializing_if = "Option::is_none")] + #[serde( + skip_serializing_if = "Option::is_none", + with = "alloy_serde::num::u128_hex_or_decimal_opt" + )] pub max_fee_per_blob_gas: Option, /// The gas limit for the transaction. + #[serde(default, with = "alloy_serde::num::u128_hex_or_decimal_opt")] pub gas: Option, /// The value transferred in the transaction, in wei. pub value: Option, From 31cdfb84bbf6809b510f72182b61403ea2343f81 Mon Sep 17 00:00:00 2001 From: Yash Atreya <44857776+yash-atreya@users.noreply.github.com> Date: Fri, 5 Apr 2024 23:56:31 -0400 Subject: [PATCH 04/12] fix(rpc-types): request tests --- crates/rpc-types/src/eth/transaction/request.rs | 1 + 1 file changed, 1 insertion(+) diff --git a/crates/rpc-types/src/eth/transaction/request.rs b/crates/rpc-types/src/eth/transaction/request.rs index fe6a264743a..0cd01f9db0e 100644 --- a/crates/rpc-types/src/eth/transaction/request.rs +++ b/crates/rpc-types/src/eth/transaction/request.rs @@ -24,6 +24,7 @@ pub struct TransactionRequest { pub max_priority_fee_per_gas: Option, /// The max fee per blob gas for EIP-4844 blob transactions. #[serde( + default, skip_serializing_if = "Option::is_none", with = "alloy_serde::num::u128_hex_or_decimal_opt" )] From 4349a140ce50c917e850993cc7f47bbad2e54990 Mon Sep 17 00:00:00 2001 From: Yash Atreya <44857776+yash-atreya@users.noreply.github.com> Date: Fri, 5 Apr 2024 23:58:25 -0400 Subject: [PATCH 05/12] fix(rpc-types): optimism tests --- crates/rpc-types/src/eth/transaction/optimism.rs | 3 +++ 1 file changed, 3 insertions(+) diff --git a/crates/rpc-types/src/eth/transaction/optimism.rs b/crates/rpc-types/src/eth/transaction/optimism.rs index 5cfcf8140c0..2d44ae4e3e7 100644 --- a/crates/rpc-types/src/eth/transaction/optimism.rs +++ b/crates/rpc-types/src/eth/transaction/optimism.rs @@ -31,6 +31,7 @@ pub struct OptimismTransactionReceiptFields { pub deposit_receipt_version: Option, /// L1 fee for the transaction #[serde( + default, skip_serializing_if = "Option::is_none", with = "alloy_serde::num::u128_hex_or_decimal_opt" )] @@ -40,12 +41,14 @@ pub struct OptimismTransactionReceiptFields { pub l1_fee_scalar: Option, /// L1 gas price for the transaction #[serde( + default, skip_serializing_if = "Option::is_none", with = "alloy_serde::num::u128_hex_or_decimal_opt" )] pub l1_gas_price: Option, /// L1 gas used for the transaction #[serde( + default, skip_serializing_if = "Option::is_none", with = "alloy_serde::num::u128_hex_or_decimal_opt" )] From 4d1eaba6cba24608200cfd2193fabdd4a9ccbb0b Mon Sep 17 00:00:00 2001 From: Yash Atreya <44857776+yash-atreya@users.noreply.github.com> Date: Mon, 8 Apr 2024 14:10:16 -0400 Subject: [PATCH 06/12] add serde attr and fix rpc-types tests --- crates/consensus/src/header.rs | 16 ++++++++-------- crates/rpc-types/src/eth/block.rs | 28 ++++++++++++++++++++++------ crates/serde/src/num.rs | 1 - 3 files changed, 30 insertions(+), 15 deletions(-) diff --git a/crates/consensus/src/header.rs b/crates/consensus/src/header.rs index 44276b2eb59..252f05168a9 100644 --- a/crates/consensus/src/header.rs +++ b/crates/consensus/src/header.rs @@ -224,14 +224,14 @@ impl Header { mem::size_of::() + // logs bloom mem::size_of::() + // difficulty mem::size_of::() + // number - mem::size_of::() + // gas limit - mem::size_of::() + // gas used + mem::size_of::() + // gas limit + mem::size_of::() + // gas used mem::size_of::() + // timestamp mem::size_of::() + // mix hash mem::size_of::() + // nonce - mem::size_of::>() + // base fee per gas - mem::size_of::>() + // blob gas used - mem::size_of::>() + // excess blob gas + mem::size_of::>() + // base fee per gas + mem::size_of::>() + // blob gas used + mem::size_of::>() + // excess blob gas mem::size_of::>() + // parent beacon block root self.extra_data.len() // extra data } @@ -396,9 +396,9 @@ impl Decodable for Header { receipts_root: Decodable::decode(buf)?, logs_bloom: Decodable::decode(buf)?, difficulty: Decodable::decode(buf)?, - number: U256::decode(buf)?.to::(), - gas_limit: U256::decode(buf)?.to::(), - gas_used: U256::decode(buf)?.to::(), + number: u64::decode(buf)?, + gas_limit: u128::decode(buf)?, + gas_used: u128::decode(buf)?, timestamp: Decodable::decode(buf)?, extra_data: Decodable::decode(buf)?, mix_hash: Decodable::decode(buf)?, diff --git a/crates/rpc-types/src/eth/block.rs b/crates/rpc-types/src/eth/block.rs index d188367154f..063b07df16b 100644 --- a/crates/rpc-types/src/eth/block.rs +++ b/crates/rpc-types/src/eth/block.rs @@ -75,10 +75,13 @@ pub struct Header { /// Difficulty pub difficulty: U256, /// Block number + #[serde(default, with = "alloy_serde::num::u64_hex_opt")] pub number: Option, /// Gas Limit + #[serde(default, with = "alloy_serde::num::u128_hex_or_decimal")] pub gas_limit: u128, /// Gas Used + #[serde(default, with = "alloy_serde::num::u128_hex_or_decimal")] pub gas_used: u128, /// Timestamp pub timestamp: U256, @@ -101,18 +104,31 @@ pub struct Header { #[serde(skip_serializing_if = "Option::is_none")] pub mix_hash: Option, /// Nonce + #[serde(default, with = "alloy_serde::num::u64_hex_opt")] pub nonce: Option, /// Base fee per unit of gas (if past London) - #[serde(skip_serializing_if = "Option::is_none")] + #[serde( + default, + skip_serializing_if = "Option::is_none", + with = "alloy_serde::num::u128_hex_or_decimal_opt" + )] pub base_fee_per_gas: Option, /// Withdrawals root hash added by EIP-4895 and is ignored in legacy headers. #[serde(skip_serializing_if = "Option::is_none")] pub withdrawals_root: Option, /// Blob gas used - #[serde(skip_serializing_if = "Option::is_none")] + #[serde( + default, + skip_serializing_if = "Option::is_none", + with = "alloy_serde::num::u128_hex_or_decimal_opt" + )] pub blob_gas_used: Option, /// Excess blob gas - #[serde(skip_serializing_if = "Option::is_none")] + #[serde( + default, + skip_serializing_if = "Option::is_none", + with = "alloy_serde::num::u128_hex_or_decimal_opt" + )] pub excess_blob_gas: Option, /// Parent beacon block root #[serde(skip_serializing_if = "Option::is_none")] @@ -1196,7 +1212,7 @@ mod tests { let serialized = serde_json::to_string(&block).unwrap(); assert_eq!( serialized, - r#"{"hash":"0x0000000000000000000000000000000000000000000000000000000000000001","parentHash":"0x0000000000000000000000000000000000000000000000000000000000000002","sha3Uncles":"0x0000000000000000000000000000000000000000000000000000000000000003","miner":"0x0000000000000000000000000000000000000004","stateRoot":"0x0000000000000000000000000000000000000000000000000000000000000005","transactionsRoot":"0x0000000000000000000000000000000000000000000000000000000000000006","receiptsRoot":"0x0000000000000000000000000000000000000000000000000000000000000007","logsBloom":"0x00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000","difficulty":"0xd","number":"0x9","gasLimit":"0xb","gasUsed":"0xa","timestamp":"0xc","totalDifficulty":"0x186a0","extraData":"0x010203","mixHash":"0x000000000000000000000000000000000000000000000000000000000000000e","nonce":"0x000000000000000f","baseFeePerGas":"0x14","withdrawalsRoot":"0x0000000000000000000000000000000000000000000000000000000000000008","uncles":["0x0000000000000000000000000000000000000000000000000000000000000011"],"transactions":["0x0000000000000000000000000000000000000000000000000000000000000012"],"size":"0x13","withdrawals":[]}"# + r#"{"hash":"0x0000000000000000000000000000000000000000000000000000000000000001","parentHash":"0x0000000000000000000000000000000000000000000000000000000000000002","sha3Uncles":"0x0000000000000000000000000000000000000000000000000000000000000003","miner":"0x0000000000000000000000000000000000000004","stateRoot":"0x0000000000000000000000000000000000000000000000000000000000000005","transactionsRoot":"0x0000000000000000000000000000000000000000000000000000000000000006","receiptsRoot":"0x0000000000000000000000000000000000000000000000000000000000000007","logsBloom":"0x00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000","difficulty":"0xd","number":"0x9","gasLimit":"0xb","gasUsed":"0xa","timestamp":"0xc","totalDifficulty":"0x186a0","extraData":"0x010203","mixHash":"0x000000000000000000000000000000000000000000000000000000000000000e","nonce":"0xf","baseFeePerGas":"0x14","withdrawalsRoot":"0x0000000000000000000000000000000000000000000000000000000000000008","uncles":["0x0000000000000000000000000000000000000000000000000000000000000011"],"transactions":["0x0000000000000000000000000000000000000000000000000000000000000012"],"size":"0x13","withdrawals":[]}"# ); let deserialized: Block = serde_json::from_str(&serialized).unwrap(); assert_eq!(block, deserialized); @@ -1238,7 +1254,7 @@ mod tests { let serialized = serde_json::to_string(&block).unwrap(); assert_eq!( serialized, - r#"{"hash":"0x0000000000000000000000000000000000000000000000000000000000000001","parentHash":"0x0000000000000000000000000000000000000000000000000000000000000002","sha3Uncles":"0x0000000000000000000000000000000000000000000000000000000000000003","miner":"0x0000000000000000000000000000000000000004","stateRoot":"0x0000000000000000000000000000000000000000000000000000000000000005","transactionsRoot":"0x0000000000000000000000000000000000000000000000000000000000000006","receiptsRoot":"0x0000000000000000000000000000000000000000000000000000000000000007","logsBloom":"0x00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000","difficulty":"0xd","number":"0x9","gasLimit":"0xb","gasUsed":"0xa","timestamp":"0xc","totalDifficulty":"0x186a0","extraData":"0x010203","mixHash":"0x000000000000000000000000000000000000000000000000000000000000000e","nonce":"0x000000000000000f","baseFeePerGas":"0x14","withdrawalsRoot":"0x0000000000000000000000000000000000000000000000000000000000000008","uncles":[],"size":"0x13"}"# + r#"{"hash":"0x0000000000000000000000000000000000000000000000000000000000000001","parentHash":"0x0000000000000000000000000000000000000000000000000000000000000002","sha3Uncles":"0x0000000000000000000000000000000000000000000000000000000000000003","miner":"0x0000000000000000000000000000000000000004","stateRoot":"0x0000000000000000000000000000000000000000000000000000000000000005","transactionsRoot":"0x0000000000000000000000000000000000000000000000000000000000000006","receiptsRoot":"0x0000000000000000000000000000000000000000000000000000000000000007","logsBloom":"0x00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000","difficulty":"0xd","number":"0x9","gasLimit":"0xb","gasUsed":"0xa","timestamp":"0xc","totalDifficulty":"0x186a0","extraData":"0x010203","mixHash":"0x000000000000000000000000000000000000000000000000000000000000000e","nonce":"0xf","baseFeePerGas":"0x14","withdrawalsRoot":"0x0000000000000000000000000000000000000000000000000000000000000008","uncles":[],"size":"0x13"}"# ); let deserialized: Block = serde_json::from_str(&serialized).unwrap(); assert_eq!(block, deserialized); @@ -1280,7 +1296,7 @@ mod tests { let serialized = serde_json::to_string(&block).unwrap(); assert_eq!( serialized, - r#"{"hash":"0x0000000000000000000000000000000000000000000000000000000000000001","parentHash":"0x0000000000000000000000000000000000000000000000000000000000000002","sha3Uncles":"0x0000000000000000000000000000000000000000000000000000000000000003","miner":"0x0000000000000000000000000000000000000004","stateRoot":"0x0000000000000000000000000000000000000000000000000000000000000005","transactionsRoot":"0x0000000000000000000000000000000000000000000000000000000000000006","receiptsRoot":"0x0000000000000000000000000000000000000000000000000000000000000007","logsBloom":"0x00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000","difficulty":"0xd","number":"0x9","gasLimit":"0xb","gasUsed":"0xa","timestamp":"0xc","totalDifficulty":"0x186a0","extraData":"0x010203","mixHash":"0x000000000000000000000000000000000000000000000000000000000000000e","nonce":"0x000000000000000f","baseFeePerGas":"0x14","uncles":["0x0000000000000000000000000000000000000000000000000000000000000011"],"transactions":["0x0000000000000000000000000000000000000000000000000000000000000012"],"size":"0x13"}"# + r#"{"hash":"0x0000000000000000000000000000000000000000000000000000000000000001","parentHash":"0x0000000000000000000000000000000000000000000000000000000000000002","sha3Uncles":"0x0000000000000000000000000000000000000000000000000000000000000003","miner":"0x0000000000000000000000000000000000000004","stateRoot":"0x0000000000000000000000000000000000000000000000000000000000000005","transactionsRoot":"0x0000000000000000000000000000000000000000000000000000000000000006","receiptsRoot":"0x0000000000000000000000000000000000000000000000000000000000000007","logsBloom":"0x00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000","difficulty":"0xd","number":"0x9","gasLimit":"0xb","gasUsed":"0xa","timestamp":"0xc","totalDifficulty":"0x186a0","extraData":"0x010203","mixHash":"0x000000000000000000000000000000000000000000000000000000000000000e","nonce":"0xf","baseFeePerGas":"0x14","uncles":["0x0000000000000000000000000000000000000000000000000000000000000011"],"transactions":["0x0000000000000000000000000000000000000000000000000000000000000012"],"size":"0x13"}"# ); let deserialized: Block = serde_json::from_str(&serialized).unwrap(); assert_eq!(block, deserialized); diff --git a/crates/serde/src/num.rs b/crates/serde/src/num.rs index 160d185a8be..71a83afbe35 100644 --- a/crates/serde/src/num.rs +++ b/crates/serde/src/num.rs @@ -243,7 +243,6 @@ where { NumberOrHexU256::deserialize(deserializer)?.try_into_u256() } - /// serde functions for handling primitive `u128` as [U128](alloy_primitives::U128) pub mod u128_hex_or_decimal { use alloy_primitives::U128; From e589d3b1c66eddc4808b814accfb7c990b32d426 Mon Sep 17 00:00:00 2001 From: Yash Atreya <44857776+yash-atreya@users.noreply.github.com> Date: Mon, 8 Apr 2024 14:17:59 -0400 Subject: [PATCH 07/12] timestamp -> `u64` in rpc-types `block` --- crates/rpc-types/src/eth/block.rs | 9 +++++---- 1 file changed, 5 insertions(+), 4 deletions(-) diff --git a/crates/rpc-types/src/eth/block.rs b/crates/rpc-types/src/eth/block.rs index 063b07df16b..4479acb5d71 100644 --- a/crates/rpc-types/src/eth/block.rs +++ b/crates/rpc-types/src/eth/block.rs @@ -84,7 +84,8 @@ pub struct Header { #[serde(default, with = "alloy_serde::num::u128_hex_or_decimal")] pub gas_used: u128, /// Timestamp - pub timestamp: U256, + #[serde(default, with = "alloy_serde::num::u64_hex")] + pub timestamp: u64, /// Total difficulty #[serde(skip_serializing_if = "Option::is_none")] pub total_difficulty: Option, @@ -1193,7 +1194,7 @@ mod tests { gas_limit: 11, extra_data: Bytes::from(vec![1, 2, 3]), logs_bloom: Bloom::default(), - timestamp: U256::from(12), + timestamp: 12, difficulty: U256::from(13), total_difficulty: Some(U256::from(100000)), mix_hash: Some(B256::with_last_byte(14)), @@ -1235,7 +1236,7 @@ mod tests { gas_limit: 11, extra_data: Bytes::from(vec![1, 2, 3]), logs_bloom: Bloom::default(), - timestamp: U256::from(12), + timestamp: 12, difficulty: U256::from(13), total_difficulty: Some(U256::from(100000)), mix_hash: Some(B256::with_last_byte(14)), @@ -1277,7 +1278,7 @@ mod tests { gas_limit: 11, extra_data: Bytes::from(vec![1, 2, 3]), logs_bloom: Bloom::default(), - timestamp: U256::from(12), + timestamp: 12, difficulty: U256::from(13), total_difficulty: Some(U256::from(100000)), mix_hash: Some(B256::with_last_byte(14)), From efc8b811b5f70fdca61a28a78feb1502b121d0c6 Mon Sep 17 00:00:00 2001 From: Yash Atreya <44857776+yash-atreya@users.noreply.github.com> Date: Mon, 8 Apr 2024 14:32:47 -0400 Subject: [PATCH 08/12] nit --- crates/serde/src/num.rs | 1 + 1 file changed, 1 insertion(+) diff --git a/crates/serde/src/num.rs b/crates/serde/src/num.rs index 71a83afbe35..160d185a8be 100644 --- a/crates/serde/src/num.rs +++ b/crates/serde/src/num.rs @@ -243,6 +243,7 @@ where { NumberOrHexU256::deserialize(deserializer)?.try_into_u256() } + /// serde functions for handling primitive `u128` as [U128](alloy_primitives::U128) pub mod u128_hex_or_decimal { use alloy_primitives::U128; From 8ca6085cc515346af37bfafd84d96673efe545e6 Mon Sep 17 00:00:00 2001 From: Yash Atreya <44857776+yash-atreya@users.noreply.github.com> Date: Mon, 8 Apr 2024 14:47:36 -0400 Subject: [PATCH 09/12] serde attr nits --- crates/eips/src/eip1559/basefee.rs | 2 ++ crates/genesis/src/lib.rs | 16 +++++++--------- 2 files changed, 9 insertions(+), 9 deletions(-) diff --git a/crates/eips/src/eip1559/basefee.rs b/crates/eips/src/eip1559/basefee.rs index a8f4b64894f..1794344507f 100644 --- a/crates/eips/src/eip1559/basefee.rs +++ b/crates/eips/src/eip1559/basefee.rs @@ -7,8 +7,10 @@ use crate::eip1559::constants::{ #[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))] pub struct BaseFeeParams { /// The base_fee_max_change_denominator from EIP-1559 + #[cfg_attr(feature = "serde", serde(with = "alloy_serde::num::u128_hex_or_decimal"))] pub max_change_denominator: u128, /// The elasticity multiplier from EIP-1559 + #[cfg_attr(feature = "serde", serde(with = "alloy_serde::num::u128_hex_or_decimal"))] pub elasticity_multiplier: u128, } diff --git a/crates/genesis/src/lib.rs b/crates/genesis/src/lib.rs index 2091478f274..745f6b8f873 100644 --- a/crates/genesis/src/lib.rs +++ b/crates/genesis/src/lib.rs @@ -24,9 +24,7 @@ use alloc::collections::BTreeMap; use alloy_primitives::{Address, Bytes, B256, U256}; use alloy_serde::{ - num::{ - u128_hex_or_decimal, u128_hex_or_decimal_opt, u64_hex_or_decimal, u64_hex_or_decimal_opt, - }, + num::{u128_hex_or_decimal, u128_hex_or_decimal_opt, u64_hex, u64_hex_or_decimal_opt}, storage::deserialize_storage_map, ttd::deserialize_json_ttd_opt, }; @@ -40,10 +38,10 @@ pub struct Genesis { #[serde(default)] pub config: ChainConfig, /// The genesis header nonce. - #[serde(with = "u64_hex_or_decimal")] + #[serde(with = "u64_hex")] pub nonce: u64, /// The genesis header timestamp. - #[serde(with = "u64_hex_or_decimal")] + #[serde(with = "u64_hex")] pub timestamp: u64, /// The genesis header extra data. pub extra_data: Bytes, @@ -66,16 +64,16 @@ pub struct Genesis { // should NOT be set in a real genesis file, but are included here for compatibility with // consensus tests, which have genesis files with these fields populated. /// The genesis header base fee - #[serde(skip_serializing_if = "Option::is_none", with = "u128_hex_or_decimal_opt")] + #[serde(default, skip_serializing_if = "Option::is_none", with = "u128_hex_or_decimal_opt")] pub base_fee_per_gas: Option, /// The genesis header excess blob gas - #[serde(skip_serializing_if = "Option::is_none", with = "u128_hex_or_decimal_opt")] + #[serde(default, skip_serializing_if = "Option::is_none", with = "u128_hex_or_decimal_opt")] pub excess_blob_gas: Option, /// The genesis header blob gas used - #[serde(skip_serializing_if = "Option::is_none", with = "u128_hex_or_decimal_opt")] + #[serde(default, skip_serializing_if = "Option::is_none", with = "u128_hex_or_decimal_opt")] pub blob_gas_used: Option, /// The genesis block number - #[serde(skip_serializing_if = "Option::is_none", with = "u64_hex_or_decimal_opt")] + #[serde(default, skip_serializing_if = "Option::is_none", with = "u64_hex_or_decimal_opt")] pub number: Option, } From 1e49002b2f753d83cad07ffad463fcf0e3b34097 Mon Sep 17 00:00:00 2001 From: Yash Atreya <44857776+yash-atreya@users.noreply.github.com> Date: Mon, 8 Apr 2024 16:18:51 -0400 Subject: [PATCH 10/12] skip_serializing nits --- crates/rpc-types/src/eth/block.rs | 6 +++++- .../rpc-types/src/eth/transaction/request.rs | 18 +++++++++++++++--- 2 files changed, 20 insertions(+), 4 deletions(-) diff --git a/crates/rpc-types/src/eth/block.rs b/crates/rpc-types/src/eth/block.rs index 4479acb5d71..ec02f8e5eb3 100644 --- a/crates/rpc-types/src/eth/block.rs +++ b/crates/rpc-types/src/eth/block.rs @@ -105,7 +105,11 @@ pub struct Header { #[serde(skip_serializing_if = "Option::is_none")] pub mix_hash: Option, /// Nonce - #[serde(default, with = "alloy_serde::num::u64_hex_opt")] + #[serde( + default, + skip_serializing_if = "Option::is_none", + with = "alloy_serde::num::u64_hex_opt" + )] pub nonce: Option, /// Base fee per unit of gas (if past London) #[serde( diff --git a/crates/rpc-types/src/eth/transaction/request.rs b/crates/rpc-types/src/eth/transaction/request.rs index 0cd01f9db0e..d2cfaa19d37 100644 --- a/crates/rpc-types/src/eth/transaction/request.rs +++ b/crates/rpc-types/src/eth/transaction/request.rs @@ -14,13 +14,25 @@ pub struct TransactionRequest { /// The destination address of the transaction. pub to: Option
, /// The legacy gas price. - #[serde(default, with = "alloy_serde::num::u128_hex_or_decimal_opt")] + #[serde( + default, + skip_serializing_if = "Option::is_none", + with = "alloy_serde::num::u128_hex_or_decimal_opt" + )] pub gas_price: Option, /// The max base fee per gas the sender is willing to pay. - #[serde(default, with = "alloy_serde::num::u128_hex_or_decimal_opt")] + #[serde( + default, + skip_serializing_if = "Option::is_none", + with = "alloy_serde::num::u128_hex_or_decimal_opt" + )] pub max_fee_per_gas: Option, /// The max priority fee per gas the sender is willing to pay, also called the miner tip. - #[serde(default, with = "alloy_serde::num::u128_hex_or_decimal_opt")] + #[serde( + default, + skip_serializing_if = "Option::is_none", + with = "alloy_serde::num::u128_hex_or_decimal_opt" + )] pub max_priority_fee_per_gas: Option, /// The max fee per blob gas for EIP-4844 blob transactions. #[serde( From 66212955655177d1fb620b13c94668aeb1565f70 Mon Sep 17 00:00:00 2001 From: Yash Atreya <44857776+yash-atreya@users.noreply.github.com> Date: Mon, 8 Apr 2024 16:42:25 -0400 Subject: [PATCH 11/12] rpc-types: make `gas_used` required in receipt --- crates/provider/src/layers/gas.rs | 4 ++-- crates/rpc-types/src/eth/transaction/receipt.rs | 4 ++-- 2 files changed, 4 insertions(+), 4 deletions(-) diff --git a/crates/provider/src/layers/gas.rs b/crates/provider/src/layers/gas.rs index 0b4d15949d3..ac2c098776c 100644 --- a/crates/provider/src/layers/gas.rs +++ b/crates/provider/src/layers/gas.rs @@ -251,7 +251,7 @@ mod tests { let tx = tx.get_receipt().await.unwrap(); assert_eq!(tx.effective_gas_price, 0x3b9aca00); - assert_eq!(tx.gas_used, Some(0x5208)); + assert_eq!(tx.gas_used, 0x5208); } #[tokio::test] @@ -281,7 +281,7 @@ mod tests { let tx = tx.get_receipt().await.unwrap(); - assert_eq!(tx.gas_used, Some(0x5208)); + assert_eq!(tx.gas_used, 0x5208); } #[tokio::test] diff --git a/crates/rpc-types/src/eth/transaction/receipt.rs b/crates/rpc-types/src/eth/transaction/receipt.rs index 7c79dd18754..1d1fde362d1 100644 --- a/crates/rpc-types/src/eth/transaction/receipt.rs +++ b/crates/rpc-types/src/eth/transaction/receipt.rs @@ -28,8 +28,8 @@ pub struct TransactionReceipt> { #[serde(with = "alloy_serde::u64_hex_opt")] pub block_number: Option, /// Gas used by this transaction alone. - #[serde(with = "alloy_serde::u128_hex_or_decimal_opt")] - pub gas_used: Option, + #[serde(with = "alloy_serde::u128_hex_or_decimal")] + pub gas_used: u128, /// The price paid post-execution by the transaction (i.e. base fee + priority fee). Both /// fields in 1559-style transactions are maximums (max fee + max priority fee), the amount /// that's actually paid by users can only be determined post-execution From fbf5623894249ab243d516a536e838ec1cbc0552 Mon Sep 17 00:00:00 2001 From: Yash Atreya <44857776+yash-atreya@users.noreply.github.com> Date: Mon, 8 Apr 2024 16:51:18 -0400 Subject: [PATCH 12/12] fix: gas filler tests --- crates/provider/src/fillers/gas.rs | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/crates/provider/src/fillers/gas.rs b/crates/provider/src/fillers/gas.rs index 95bba241776..62f717f3e5a 100644 --- a/crates/provider/src/fillers/gas.rs +++ b/crates/provider/src/fillers/gas.rs @@ -276,7 +276,7 @@ mod tests { let tx = tx.get_receipt().await.unwrap(); assert_eq!(tx.effective_gas_price, 0x3b9aca00); - assert_eq!(tx.gas_used, Some(0x5208)); + assert_eq!(tx.gas_used, 0x5208); } #[tokio::test] @@ -297,7 +297,7 @@ mod tests { let receipt = tx.get_receipt().await.unwrap(); - assert_eq!(receipt.gas_used, Some(0x5208)); + assert_eq!(receipt.gas_used, 0x5208); } #[tokio::test]