Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: Add Unit Tests for Mempool Functionality #1359

Merged
merged 8 commits into from
Sep 9, 2024
Merged
Show file tree
Hide file tree
Changes from 3 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
396 changes: 396 additions & 0 deletions tests/tests/mempool.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,396 @@
#![allow(clippy::used_underscore_binding)]
#![cfg(feature = "testing")]
use kakarot_rpc::{
providers::eth_provider::{error::SignatureError, ChainProvider},
test_utils::{
eoa::Eoa,
fixtures::{katana, setup},
katana::Katana,
},
};
use reth_primitives::{
sign_message, Address, Bytes, Transaction, TransactionSigned, TransactionSignedEcRecovered, TxEip1559, TxKind, U256,
};
use reth_transaction_pool::{EthPooledTransaction, TransactionOrigin, TransactionPool};
use rstest::*;

#[rstest]
#[awt]
#[tokio::test(flavor = "multi_thread")]
async fn test_mempool_add_transaction(#[future] katana: Katana, _setup: ()) {
let eth_provider = katana.eth_provider();

// Create a sample transaction
let (transaction, transaction_signed) = create_sample_transactions(&katana, 1)
.await
.expect("Failed to create sample transaction")
.pop()
.expect("Expected at least one transaction");

// Check initial pool size
assert_eq!(eth_provider.mempool().unwrap().pool_size().total, 0);

// Add transaction to mempool
let result = eth_provider.mempool().unwrap().add_transaction(TransactionOrigin::Local, transaction.clone()).await;

// Ensure the transaction was added successfully
assert!(result.is_ok());

// Get updated mempool size
let mempool_size = eth_provider.mempool().unwrap().pool_size();

// Get the EOA address
let address = katana.eoa().evm_address().expect("Failed to get eoa address");

// Get transactions by sender address and nonce
let sender_transactions = eth_provider.mempool().unwrap().get_transactions_by_sender_and_nonce(address, 0);

// Get transactions by origin
let origin_transaction = eth_provider.mempool().unwrap().get_transactions_by_origin(TransactionOrigin::Local);

// Get local transactions
let local_transaction = eth_provider.mempool().unwrap().get_local_transactions();

// Get all transactions in the mempool
let all_transactions = eth_provider.mempool().unwrap().all_transactions();

// Check pending, queued and total transactions
assert_eq!(mempool_size.pending, 1);
assert_eq!(mempool_size.queued, 0);
assert_eq!(mempool_size.total, 1);

// get_transactions_by_sender_and_nonce test
// Check if the returned transaction hash matches
assert_eq!(*sender_transactions.unwrap().hash(), transaction_signed.hash());

// get_transactions_by_origin function test
// Check if the returned transaction hash matches
assert_eq!(*origin_transaction[0].hash(), transaction_signed.hash());

// get_local_transactions function test
// Check if the returned transaction hash matches
assert_eq!(*local_transaction[0].hash(), transaction_signed.hash());
assert_eq!(*local_transaction[0].hash(), *origin_transaction[0].hash());

// Remove transaction by hash
let _ = eth_provider.mempool().unwrap().remove_transactions(vec![transaction_signed.hash()]);

// remove_transactions function tests
// Get updated mempool size
let mempool_size = eth_provider.mempool().unwrap().pool_size();
// Check pending, queued and total transactions after remove_transactions
assert_eq!(mempool_size.pending, 0);
assert_eq!(mempool_size.queued, 0);
assert_eq!(mempool_size.total, 0);

// all_transactions function tests
// Check if the first pending transaction hash matches
assert_eq!(*all_transactions.pending[0].hash(), transaction_signed.hash());
// Ensure only one pending transaction is present
assert_eq!(all_transactions.pending.len(), 1);
// Ensure no queued transactions are present
assert_eq!(all_transactions.queued.len(), 0);
eugypalu marked this conversation as resolved.
Show resolved Hide resolved
}

#[rstest]
#[awt]
#[tokio::test(flavor = "multi_thread")]
async fn test_mempool_add_external_transaction(#[future] katana: Katana, _setup: ()) {
let eth_provider = katana.eth_provider();

// Create a sample transaction
let (transaction, transaction_signed) = create_sample_transactions(&katana, 1)
.await
.expect("Failed to create sample transaction")
.pop()
.expect("Expected at least one transaction");

// Add external transaction
let result = eth_provider.mempool().unwrap().add_external_transaction(transaction).await;

// Get pooled transaction by hash
let hashes = eth_provider.mempool().unwrap().get_pooled_transaction_element(transaction_signed.hash());

// Ensure the transaction was added successfully
assert!(result.is_ok());

// Get updated mempool size
let mempool_size = eth_provider.mempool().unwrap().pool_size();

// Check pending, queued and total transactions
assert_eq!(mempool_size.pending, 1);
assert_eq!(mempool_size.queued, 0);
assert_eq!(mempool_size.total, 1);

// get_pooled_transaction_element function test
// Check if the retrieved hash matches the expected hash
assert_eq!(hashes.unwrap().hash(), &transaction_signed.hash());
}

#[rstest]
#[awt]
#[tokio::test(flavor = "multi_thread")]
async fn test_mempool_add_transactions(#[future] katana: Katana, _setup: ()) {
let eth_provider = katana.eth_provider();
// Get the EOA address
let address = katana.eoa().evm_address().expect("Failed to get eoa address");

// Set the number of transactions to create
let transaction_number = 2;

// Create multiple sample transactions
let transactions =
create_sample_transactions(&katana, transaction_number).await.expect("Failed to create sample transaction");

// Collect pooled transactions
let pooled_transactions =
transactions.iter().map(|(eth_pooled_transaction, _)| eth_pooled_transaction.clone()).collect::<Vec<_>>();

// Collect signed transactions
let signed_transactions =
transactions.iter().map(|(_, signed_transactions)| signed_transactions.clone()).collect::<Vec<_>>();

// Add transactions to mempool
let _ = eth_provider.mempool().unwrap().add_transactions(TransactionOrigin::Local, pooled_transactions).await;

// Get pending transactions
let hashes = eth_provider.mempool().unwrap().pending_transactions();

// Get transactions by sender address
let sender_transactions = eth_provider.mempool().unwrap().get_transactions_by_sender(address);

// Get unique senders from the mempool
let unique_senders = eth_provider.mempool().unwrap().unique_senders();

// Check if the first signed transaction is contained
let contains = eth_provider.mempool().unwrap().contains(&signed_transactions[0].hash());

// Get updated mempool size
let mempool_size = eth_provider.mempool().unwrap().pool_size();
// mempool_size function tests
// Check pending transactions
assert_eq!(mempool_size.pending, transaction_number);
// Check queued transactions
assert_eq!(mempool_size.queued, 0);
// Check total transactions
assert_eq!(mempool_size.total, transaction_number);

// pending_transactions function tests
// Check if the first pending transaction hash matches
assert_eq!(hashes[0].hash(), &signed_transactions[0].hash());
// Check if the second pending transaction hash matches
assert_eq!(hashes[1].hash(), &signed_transactions[1].hash());
// Ensure the number of pending transactions matches the expected count
assert_eq!(hashes.len(), transaction_number);

// get_transactions_by_sender function tests
// Ensure only one transaction is returned
assert_eq!(sender_transactions.len(), transaction_number);
// Check if the returned transaction hash matches
assert_eq!(*sender_transactions[0].hash(), signed_transactions[0].hash());
assert_eq!(*sender_transactions[1].hash(), signed_transactions[1].hash());
eugypalu marked this conversation as resolved.
Show resolved Hide resolved

// unique_senders function test
// Ensure the EOA address is in the unique senders
assert!(unique_senders.contains(&address));

// contains function test
assert!(contains);
}

#[rstest]
#[awt]
#[tokio::test(flavor = "multi_thread")]
async fn test_mempool_add_external_transactions(#[future] katana: Katana, _setup: ()) {
let eth_provider = katana.eth_provider();

// Create multiple sample transactions
let transactions = create_sample_transactions(&katana, 2).await.expect("Failed to create sample transaction");

// Collect pooled transactions
let pooled_transactions =
transactions.iter().map(|(eth_pooled_transaction, _)| eth_pooled_transaction.clone()).collect::<Vec<_>>();

// Collect signed transactions
let signed_transactions =
transactions.iter().map(|(_, signed_transactions)| signed_transactions.clone()).collect::<Vec<_>>();

// Add external transactions to mempool
let _ = eth_provider.mempool().unwrap().add_external_transactions(pooled_transactions).await;

// Get pooled transaction hashes
let hashes = eth_provider.mempool().unwrap().pooled_transaction_hashes();

// Set maximum number of hashes to retrieve
let hashes_max_number = 1;

// Get pooled transaction hashes with a limit
let hashes_max = eth_provider.mempool().unwrap().pooled_transaction_hashes_max(hashes_max_number);

// Get external transactions
let external_transactions = eth_provider.mempool().unwrap().get_external_transactions();

// Get updated mempool size
let mempool_size = eth_provider.mempool().unwrap().pool_size();
// Check pending transactions
assert_eq!(mempool_size.pending, 2);
// Check queued transactions
assert_eq!(mempool_size.queued, 0);
// Check total transactions
assert_eq!(mempool_size.total, 2);

// pooled_transaction_hashes function tests
// Check if the first signed transaction hash is present
assert!(hashes.contains(&signed_transactions[0].hash()));
// Check if the second signed transaction hash is present
assert!(hashes.contains(&signed_transactions[1].hash()));
// Ensure the hashes are not empty
assert!(!hashes.is_empty());

// pooled_transaction_hashes_max function test
// Check if at least one signed transaction hash is present
assert!(hashes_max.contains(&signed_transactions[0].hash()) || hashes_max.contains(&signed_transactions[1].hash()));
// Ensure the number of hashes matches the limit
assert_eq!(hashes_max.len(), hashes_max_number);
// Ensure the hashes are not empty
assert!(!hashes_max.is_empty());
eugypalu marked this conversation as resolved.
Show resolved Hide resolved

// get_external_transactions function test
// Check if the returned transaction hash matches
assert_eq!(*external_transactions[0].hash(), signed_transactions[0].hash());
assert_eq!(*external_transactions[1].hash(), signed_transactions[1].hash());
}

#[rstest]
#[awt]
#[tokio::test(flavor = "multi_thread")]
async fn test_mempool_add_transaction_and_subscribe(#[future] katana: Katana, _setup: ()) {
eugypalu marked this conversation as resolved.
Show resolved Hide resolved
let eth_provider = katana.eth_provider();

// Create a sample transaction
let (transaction, _) = create_sample_transactions(&katana, 1)
.await
.expect("Failed to create sample transaction")
.pop()
.expect("Expected at least one transaction");

// Add transaction and subscribe to events
let result = eth_provider
.mempool()
.unwrap()
.add_transaction_and_subscribe(TransactionOrigin::Local, transaction.clone())
.await;

// Ensure the transaction was added successfully
assert!(result.is_ok());

// Get updated mempool size
let mempool_size = eth_provider.mempool().unwrap().pool_size();
// Check pending transactions
assert_eq!(mempool_size.pending, 1);
// Check queued transactions
assert_eq!(mempool_size.queued, 0);
// Check total transactions
assert_eq!(mempool_size.total, 1);
}

#[rstest]
#[awt]
#[tokio::test(flavor = "multi_thread")]
async fn test_mempool_transaction_event_listener(#[future] katana: Katana, _setup: ()) {
let eth_provider = katana.eth_provider();

// Create a sample transaction
let (transaction, transaction_signed) = create_sample_transactions(&katana, 1)
.await
.expect("Failed to create sample transaction")
.pop()
.expect("Expected at least one transaction");

// Add transaction to mempool
eth_provider.mempool().unwrap().add_transaction(TransactionOrigin::Local, transaction.clone()).await.unwrap();

// Get the transaction event listener
let listener = eth_provider.mempool().unwrap().transaction_event_listener(transaction_signed.hash());

// Ensure the listener exists
assert!(listener.is_some());

// Check if the listener's hash matches the transaction's hash
assert_eq!(listener.unwrap().hash(), transaction_signed.hash());
}

#[rstest]
#[awt]
#[tokio::test(flavor = "multi_thread")]
async fn test_mempool_get_private_transactions(#[future] katana: Katana, _setup: ()) {
let eth_provider = katana.eth_provider();

// Create a sample transaction
let (transaction, transaction_signed) = create_sample_transactions(&katana, 1)
.await
.expect("Failed to create sample transaction")
.pop()
.expect("Expected at least one transaction");

// Add private transaction to mempool
eth_provider.mempool().unwrap().add_transaction(TransactionOrigin::Private, transaction.clone()).await.unwrap();

// Get private transactions
let private_transaction = eth_provider.mempool().unwrap().get_private_transactions();

// Check if the returned transaction hash matches
assert_eq!(*private_transaction[0].hash(), transaction_signed.hash());
}

// Helper function to create a sample transaction
async fn create_sample_transactions(
katana: &Katana,
num_transactions: usize,
) -> Result<Vec<(EthPooledTransaction, TransactionSigned)>, SignatureError> {
// Initialize a vector to hold transactions
let mut transactions = Vec::new();

for counter in 0..num_transactions {
// Get the Ethereum provider
let eth_provider = katana.eth_provider();

// Get the chain ID
let chain_id = eth_provider.chain_id().await.unwrap_or_default().unwrap_or_default().to();
eugypalu marked this conversation as resolved.
Show resolved Hide resolved

// Create a new EIP-1559 transaction
let transaction = Transaction::Eip1559(TxEip1559 {
chain_id,
nonce: counter as u64,
gas_limit: 21000,
to: TxKind::Call(Address::random()),
value: U256::from(1000),
input: Bytes::default(),
max_fee_per_gas: 875_000_000,
max_priority_fee_per_gas: 0,
access_list: Default::default(),
eugypalu marked this conversation as resolved.
Show resolved Hide resolved
});

// Sign the transaction
let signature = sign_message(katana.eoa().private_key(), transaction.signature_hash()).unwrap();

// Create a signed transaction
let transaction_signed = TransactionSigned::from_transaction_and_signature(transaction, signature);
// Recover the signer from the transaction
let signer = transaction_signed.recover_signer().ok_or(SignatureError::Recovery)?;
eugypalu marked this conversation as resolved.
Show resolved Hide resolved

// Create an EC recovered signed transaction
let transaction_signed_ec_recovered =
TransactionSignedEcRecovered::from_signed_transaction(transaction_signed.clone(), signer);

// Get the encoded length of the transaction
let encoded_length = transaction_signed_ec_recovered.clone().length_without_header();

// Create a pooled transaction
let eth_pooled_transaction = EthPooledTransaction::new(transaction_signed_ec_recovered, encoded_length);

// Add the transaction to the vector
transactions.push((eth_pooled_transaction, transaction_signed));
}
Ok(transactions)
}
Loading
Loading