Files
reth/crates/transaction-pool/src/identifier.rs
2024-02-08 01:57:32 +00:00

147 lines
4.5 KiB
Rust

use fnv::FnvHashMap;
use reth_primitives::Address;
use std::collections::HashMap;
/// An internal mapping of addresses.
///
/// This assigns a _unique_ `SenderId` for a new `Address`.
/// It has capacity for 2^64 unique addresses.
#[derive(Debug, Default)]
pub(crate) struct SenderIdentifiers {
/// The identifier to use next.
id: u64,
/// Assigned `SenderId` for an `Address`.
address_to_id: HashMap<Address, SenderId>,
/// Reverse mapping of `SenderId` to `Address`.
sender_to_address: FnvHashMap<SenderId, Address>,
}
impl SenderIdentifiers {
/// Returns the address for the given identifier.
#[allow(dead_code)]
pub(crate) fn address(&self, id: &SenderId) -> Option<&Address> {
self.sender_to_address.get(id)
}
/// Returns the `SenderId` that belongs to the given address, if it exists
pub(crate) fn sender_id(&self, addr: &Address) -> Option<SenderId> {
self.address_to_id.get(addr).copied()
}
/// Returns the existing `SendId` or assigns a new one if it's missing
pub(crate) fn sender_id_or_create(&mut self, addr: Address) -> SenderId {
self.sender_id(&addr).unwrap_or_else(|| {
let id = self.next_id();
self.address_to_id.insert(addr, id);
self.sender_to_address.insert(id, addr);
id
})
}
/// Returns a new address
fn next_id(&mut self) -> SenderId {
let id = self.id;
self.id = self.id.wrapping_add(1);
SenderId(id)
}
}
/// A _unique_ identifier for a sender of an address.
///
/// This is the identifier of an internal `address` mapping that is valid in the context of this
/// program.
#[derive(Debug, Copy, Clone, Eq, PartialEq, Ord, PartialOrd, Hash)]
pub struct SenderId(u64);
// === impl SenderId ===
impl SenderId {
/// Returns a `Bound` for `TransactionId` starting with nonce `0`
pub(crate) const fn start_bound(self) -> std::ops::Bound<TransactionId> {
std::ops::Bound::Included(TransactionId::new(self, 0))
}
}
impl From<u64> for SenderId {
fn from(value: u64) -> Self {
SenderId(value)
}
}
/// A unique identifier of a transaction of a Sender.
///
/// This serves as an identifier for dependencies of a transaction:
/// A transaction with a nonce higher than the current state nonce depends on `tx.nonce - 1`.
#[derive(Debug, Copy, Clone, Eq, PartialEq, Ord, PartialOrd, Hash)]
pub struct TransactionId {
/// Sender of this transaction
pub sender: SenderId,
/// Nonce of this transaction
pub nonce: u64,
}
// === impl TransactionId ===
impl TransactionId {
/// Create a new identifier pair
pub const fn new(sender: SenderId, nonce: u64) -> Self {
Self { sender, nonce }
}
/// Returns the `TransactionId` this transaction depends on.
///
/// This returns `transaction_nonce - 1` if `transaction_nonce` is higher than the
/// `on_chain_nonce`
pub fn ancestor(
transaction_nonce: u64,
on_chain_nonce: u64,
sender: SenderId,
) -> Option<TransactionId> {
if transaction_nonce == on_chain_nonce {
return None
}
let prev_nonce = transaction_nonce.saturating_sub(1);
(on_chain_nonce <= prev_nonce).then(|| Self::new(sender, prev_nonce))
}
/// Returns the `TransactionId` that would come before this transaction.
pub(crate) fn unchecked_ancestor(&self) -> Option<TransactionId> {
(self.nonce != 0).then(|| TransactionId::new(self.sender, self.nonce - 1))
}
/// Returns the `TransactionId` that directly follows this transaction: `self.nonce + 1`
pub const fn descendant(&self) -> TransactionId {
TransactionId::new(self.sender, self.nonce + 1)
}
/// Returns the nonce that follows immediately after this one.
#[inline]
pub(crate) const fn next_nonce(&self) -> u64 {
self.nonce + 1
}
}
#[cfg(test)]
mod tests {
use super::*;
use std::collections::BTreeSet;
#[test]
fn test_transaction_id_ord_eq_sender() {
let tx1 = TransactionId::new(100u64.into(), 0u64);
let tx2 = TransactionId::new(100u64.into(), 1u64);
assert!(tx2 > tx1);
let set = BTreeSet::from([tx1, tx2]);
assert_eq!(set.into_iter().collect::<Vec<_>>(), vec![tx1, tx2]);
}
#[test]
fn test_transaction_id_ord() {
let tx1 = TransactionId::new(99u64.into(), 0u64);
let tx2 = TransactionId::new(100u64.into(), 1u64);
assert!(tx2 > tx1);
let set = BTreeSet::from([tx1, tx2]);
assert_eq!(set.into_iter().collect::<Vec<_>>(), vec![tx1, tx2]);
}
}