diff --git a/crates/provider/src/fillers/nonce.rs b/crates/provider/src/fillers/nonce.rs
index 237cb7d95836..5f281fc31d3d 100644
--- a/crates/provider/src/fillers/nonce.rs
+++ b/crates/provider/src/fillers/nonce.rs
@@ -7,8 +7,10 @@ use alloy_network::{Network, TransactionBuilder};
use alloy_primitives::Address;
use alloy_transport::{Transport, TransportResult};
use dashmap::DashMap;
-use std::sync::Arc;
-use tokio::sync::Mutex;
+use std::sync::{
+ atomic::{AtomicU64, Ordering},
+ Arc,
+};
/// A [`TxFiller`] that fills nonces on transactions.
///
@@ -41,7 +43,7 @@ use tokio::sync::Mutex;
/// ```
#[derive(Clone, Debug, Default)]
pub struct NonceFiller {
- nonces: DashMap
>>>,
+ nonces: DashMap>,
}
impl TxFiller for NonceFiller {
@@ -92,33 +94,57 @@ impl NonceFiller {
N: Network,
T: Transport + Clone,
{
- // locks dashmap internally for a short duration to clone the `Arc`
- let mutex = Arc::clone(self.nonces.entry(from).or_default().value());
-
- // locks the value (does not lock dashmap)
- let mut nonce = mutex.lock().await;
- match *nonce {
- Some(ref mut nonce) => {
- *nonce += 1;
- Ok(*nonce)
- }
- None => {
- // initialize the nonce if we haven't seen this account before
- let initial_nonce = provider.get_transaction_count(from).await?;
- *nonce = Some(initial_nonce);
- Ok(initial_nonce)
- }
+ // Use `u64::MAX` as a sentinel value to indicate that the nonce has not been fetched yet.
+ const NONE: u64 = u64::MAX;
+
+ // Locks dashmap internally for a short duration to clone the `Arc`.
+ // We also don't want to hold the dashmap lock through the await point below.
+ let nonce = Arc::clone(
+ self.nonces.entry(from).or_insert_with(|| Arc::new(AtomicU64::new(NONE))).value(),
+ );
+
+ let prev_nonce = nonce.fetch_add(1, Ordering::AcqRel);
+ let current_nonce;
+ if prev_nonce == NONE {
+ // Initialize the nonce if we haven't seen this account before.
+ current_nonce = provider.get_transaction_count(from).await?;
+ nonce.store(current_nonce, Ordering::Release);
+ } else {
+ current_nonce = prev_nonce + 1;
+ // `nonce` is already incremented in the `fetch_add` call above.
}
+ Ok(current_nonce)
}
}
#[cfg(test)]
mod tests {
use super::*;
- use crate::{ProviderBuilder, WalletProvider};
+ use crate::{ext::AnvilApi, ProviderBuilder, WalletProvider};
use alloy_primitives::{address, U256};
use alloy_rpc_types_eth::TransactionRequest;
+ #[tokio::test]
+ async fn smoke_test() {
+ let filler = NonceFiller::default();
+ let provider = ProviderBuilder::new().on_anvil();
+ let address = Address::ZERO;
+ for i in 0..5 {
+ let nonce = filler.get_next_nonce(&provider, address).await.unwrap();
+ assert_eq!(nonce, i);
+ }
+
+ #[cfg(feature = "anvil-api")]
+ {
+ filler.nonces.clear();
+ provider.anvil_set_nonce(address, U256::from(69)).await.unwrap();
+ for i in 0..5 {
+ let nonce = filler.get_next_nonce(&provider, address).await.unwrap();
+ assert_eq!(nonce, 69 + i);
+ }
+ }
+ }
+
#[tokio::test]
async fn no_nonce_if_sender_unset() {
let provider = ProviderBuilder::new().with_nonce_management().on_anvil();