Skip to content

Commit

Permalink
refactor: update priority queue to use PrioritizedTransaction
Browse files Browse the repository at this point in the history
  • Loading branch information
MohammadNassar1 committed Jun 27, 2024
1 parent 8f12d8d commit 6249321
Show file tree
Hide file tree
Showing 4 changed files with 46 additions and 41 deletions.
36 changes: 23 additions & 13 deletions crates/mempool/src/mempool.rs
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
use std::collections::HashMap;

use starknet_api::core::ContractAddress;
use starknet_api::transaction::TransactionHash;
use starknet_api::core::{ContractAddress, Nonce};
use starknet_api::transaction::{Tip, TransactionHash};
use starknet_mempool_types::mempool_types::{
AccountState, MempoolInput, MempoolResult, ThinTransaction,
};
Expand All @@ -16,8 +16,10 @@ pub mod mempool_test;
#[derive(Debug, Default)]
pub struct Mempool {
// TODO: add docstring explaining visibility and coupling of the fields.
txs_queue: TransactionQueue,
// All transactions currently held in the mempool.
tx_pool: TransactionPool,
// Transactions eligible for sequencing.
tx_queue: TransactionQueue,
}

impl Mempool {
Expand All @@ -41,11 +43,9 @@ impl Mempool {
// library.
pub fn get_txs(&mut self, n_txs: usize) -> MempoolResult<Vec<ThinTransaction>> {
let mut eligible_txs: Vec<ThinTransaction> = Vec::with_capacity(n_txs);

let txs = self.txs_queue.pop_last_chunk(n_txs);
for tx in txs {
self.tx_pool.remove(tx.tx_hash)?;
eligible_txs.push(tx.0);
for tx_hash in self.tx_queue.pop_last_chunk(n_txs) {
let tx = self.tx_pool.remove(tx_hash)?;
eligible_txs.push(tx);
}

Ok(eligible_txs)
Expand Down Expand Up @@ -75,7 +75,7 @@ impl Mempool {
let tx = input.tx;

self.tx_pool.insert(tx.clone())?;
self.txs_queue.insert(TransactionReference::new(tx));
self.tx_queue.insert(TransactionReference::new(&tx));

Ok(())
}
Expand All @@ -85,11 +85,21 @@ impl Mempool {
/// execution fields).
/// TODO(Mohammad): rename this struct to `ThinTransaction` once that name
/// becomes available, to better reflect its purpose and usage.
#[derive(Clone, Debug, Default, derive_more::Deref)]
pub struct TransactionReference(pub ThinTransaction);
#[derive(Clone, Copy, Debug, Default, Eq, PartialEq)]
pub struct TransactionReference {
pub sender_address: ContractAddress,
pub nonce: Nonce,
pub tx_hash: TransactionHash,
pub tip: Tip,
}

impl TransactionReference {
pub fn new(tx: ThinTransaction) -> Self {
TransactionReference(tx)
pub fn new(tx: &ThinTransaction) -> Self {
TransactionReference {
sender_address: tx.sender_address,
nonce: tx.nonce,
tx_hash: tx.tx_hash,
tip: tx.tip,
}
}
}
7 changes: 4 additions & 3 deletions crates/mempool/src/mempool_test.rs
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@ use starknet_api::{contract_address, patricia_key};
use starknet_mempool_types::errors::MempoolError;
use starknet_mempool_types::mempool_types::{Account, ThinTransaction};

use crate::mempool::{Mempool, MempoolInput};
use crate::mempool::{Mempool, MempoolInput, TransactionReference};

/// Creates a valid input for mempool's `add_tx` with optional default value for
/// `sender_address`.
Expand Down Expand Up @@ -48,12 +48,13 @@ fn mempool() -> Mempool {
// transactions.
#[track_caller]
fn check_mempool_txs_eq(mempool: &Mempool, expected_txs: &[ThinTransaction]) {
let mempool_txs = mempool.txs_queue.iter();
let mempool_txs = mempool.tx_queue.iter();
let expected_txs = expected_txs.iter().map(TransactionReference::new);

assert!(
zip_eq(expected_txs, mempool_txs)
// Deref the inner mempool tx type.
.all(|(expected_tx, mempool_tx)| *expected_tx == **mempool_tx)
.all(|(expected_tx, mempool_tx)| expected_tx == *mempool_tx)
);
}

Expand Down
26 changes: 13 additions & 13 deletions crates/mempool/src/transaction_pool.rs
Original file line number Diff line number Diff line change
Expand Up @@ -20,29 +20,29 @@ pub struct TransactionPool {
}

impl TransactionPool {
// TODO(Mohammad): Remove the cloning of tx once the `TransactionReference` is updated.
pub fn insert(&mut self, tx: ThinTransaction) -> MempoolResult<()> {
let tx_hash = tx.tx_hash;

// Insert transaction to pool, if it is new.
if let hash_map::Entry::Vacant(entry) = self.tx_pool.entry(tx_hash) {
entry.insert(tx.clone());
} else {
return Err(MempoolError::DuplicateTransaction { tx_hash });
}

// Insert transaction into the mapping by account.
let txs_from_account_entry = self.txs_by_account.entry(tx.sender_address).or_default();
match txs_from_account_entry.entry(tx.nonce) {
btree_map::Entry::Vacant(txs_from_account) => {
txs_from_account.insert(TransactionReference::new(tx));
txs_from_account.insert(TransactionReference::new(&tx));
}
btree_map::Entry::Occupied(_) => {
panic!(
"Transaction pool consistency error: transaction with hash {tx_hash} does not \
appear in main mapping, but it appears in the account mapping"
);
return Err(MempoolError::DuplicateTransaction { tx_hash });
}
}

// Insert transaction to pool, if it is new.
if let hash_map::Entry::Vacant(entry) = self.tx_pool.entry(tx_hash) {
entry.insert(tx);
} else {
panic!(
"Transaction pool consistency error: transaction with hash {tx_hash} does not \
appear in the account mapping, but it appears in the main mapping"
);
}
Ok(())
}

Expand Down
18 changes: 6 additions & 12 deletions crates/mempool/src/transaction_queue.rs
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
use std::cmp::Ordering;
use std::collections::{BTreeSet, VecDeque};

use starknet_api::transaction::TransactionHash;
use starknet_mempool_types::mempool_types::ThinTransaction;

use crate::mempool::TransactionReference;
Expand All @@ -15,13 +16,12 @@ impl TransactionQueue {
/// Adds a transaction to the mempool, ensuring unique keys.
/// Panics: if given a duplicate tx.
pub fn insert(&mut self, tx: TransactionReference) {
let mempool_tx = QueuedTransaction(tx);
assert!(self.0.insert(mempool_tx), "Keys should be unique; duplicates are checked prior.");
assert!(self.0.insert(tx.into()), "Keys should be unique; duplicates are checked prior.");
}

// TODO(gilad): remove collect
pub fn pop_last_chunk(&mut self, n_txs: usize) -> Vec<TransactionReference> {
(0..n_txs).filter_map(|_| self.0.pop_last().map(|tx| tx.0)).collect()
pub fn pop_last_chunk(&mut self, n_txs: usize) -> Vec<TransactionHash> {
(0..n_txs).filter_map(|_| self.0.pop_last().map(|tx| tx.tx_hash)).collect()
}

#[cfg(any(feature = "testing", test))]
Expand All @@ -30,14 +30,8 @@ impl TransactionQueue {
}
}

impl From<Vec<TransactionReference>> for TransactionQueue {
fn from(transactions: Vec<TransactionReference>) -> Self {
TransactionQueue(BTreeSet::from_iter(transactions.into_iter().map(QueuedTransaction)))
}
}

#[derive(Clone, Debug, derive_more::Deref, derive_more::From)]
struct QueuedTransaction(pub TransactionReference);
#[derive(Clone, Debug, Default, derive_more::Deref, derive_more::DerefMut, derive_more::From)]
struct QueuedTransaction(TransactionReference);

/// Compare transactions based only on their tip, a uint, using the Eq trait. It ensures that two
/// tips are either exactly equal or not.
Expand Down

0 comments on commit 6249321

Please sign in to comment.