#![cfg_attr(docsrs, feature(doc_cfg))] #![deny(unsafe_code)] #![deny(rustdoc::broken_intra_doc_links)] #![allow(clippy::type_complexity)] #![doc = include_str!("../README.md")] mod transports; pub use transports::*; mod provider; pub use provider::{is_local_endpoint, FilterKind, Provider, ProviderError, ProviderExt}; // types for the admin api pub mod admin; pub use admin::{NodeInfo, PeerInfo}; // ENS support pub mod ens; mod pending_transaction; pub use pending_transaction::PendingTransaction; mod pending_escalator; pub use pending_escalator::EscalatingPending; mod log_query; pub use log_query::{LogQuery, LogQueryError}; mod stream; pub use futures_util::StreamExt; pub use stream::{ interval, FilterWatcher, TransactionStream, DEFAULT_LOCAL_POLL_INTERVAL, DEFAULT_POLL_INTERVAL, }; mod pubsub; pub use pubsub::{PubsubClient, SubscriptionStream}; pub mod call_raw; pub mod erc; use async_trait::async_trait; use auto_impl::auto_impl; use ethers_core::types::{ transaction::{eip2718::TypedTransaction, eip2930::AccessListWithGasUsed}, *, }; use futures_util::future::join_all; use serde::{de::DeserializeOwned, Serialize}; use std::{error::Error, fmt::Debug, future::Future, pin::Pin}; use url::Url; // feature-enabled support for dev-rpc methods #[cfg(feature = "dev-rpc")] pub use provider::dev_rpc::DevRpcMiddleware; /// A simple gas escalation policy pub type EscalationPolicy = Box U256 + Send + Sync>; // Helper type alias #[cfg(target_arch = "wasm32")] pub(crate) type PinBoxFut<'a, T> = Pin> + 'a>>; #[cfg(not(target_arch = "wasm32"))] pub(crate) type PinBoxFut<'a, T> = Pin> + Send + 'a>>; #[cfg_attr(target_arch = "wasm32", async_trait(?Send))] #[cfg_attr(not(target_arch = "wasm32"), async_trait)] #[auto_impl(&, Box, Arc)] /// Trait which must be implemented by data transports to be used with the Ethereum /// JSON-RPC provider. pub trait JsonRpcClient: Debug + Send + Sync { /// A JSON-RPC Error type Error: Error + Into; /// Sends a request with the provided JSON-RPC and parameters serialized as JSON async fn request(&self, method: &str, params: T) -> Result where T: Debug + Serialize + Send + Sync, R: DeserializeOwned + Send; } pub trait FromErr { fn from(src: T) -> Self; } /// Calls the future if `item` is None, otherwise returns a `futures::ok` pub async fn maybe(item: Option, f: F) -> Result where F: Future>, { if let Some(item) = item { futures_util::future::ok(item).await } else { f.await } } /// A middleware allows customizing requests send and received from an ethereum node. /// /// Writing a middleware is as simple as: /// 1. implementing the [`inner`](crate::Middleware::inner) method to point to the next layer in the /// "middleware onion", 2. implementing the [`FromErr`](crate::FromErr) trait on your middleware's /// error type 3. implementing any of the methods you want to override /// /// ```rust /// use ethers_providers::{Middleware, FromErr}; /// use ethers_core::types::{U64, TransactionRequest, U256, transaction::eip2718::TypedTransaction, BlockId}; /// use thiserror::Error; /// use async_trait::async_trait; /// /// #[derive(Debug)] /// struct MyMiddleware(M); /// /// #[derive(Error, Debug)] /// pub enum MyError { /// #[error("{0}")] /// MiddlewareError(M::Error), /// /// // Add your middleware's specific errors here /// } /// /// impl FromErr for MyError { /// fn from(src: M::Error) -> MyError { /// MyError::MiddlewareError(src) /// } /// } /// /// #[async_trait] /// impl Middleware for MyMiddleware /// where /// M: Middleware, /// { /// type Error = MyError; /// type Provider = M::Provider; /// type Inner = M; /// /// fn inner(&self) -> &M { /// &self.0 /// } /// /// /// Overrides the default `get_block_number` method to always return 0 /// async fn get_block_number(&self) -> Result { /// Ok(U64::zero()) /// } /// /// /// Overrides the default `estimate_gas` method to log that it was called, /// /// before forwarding the call to the next layer. /// async fn estimate_gas(&self, tx: &TypedTransaction, block: Option) -> Result { /// println!("Estimating gas..."); /// self.inner().estimate_gas(tx, block).await.map_err(FromErr::from) /// } /// } /// ``` #[cfg_attr(target_arch = "wasm32", async_trait(?Send))] #[cfg_attr(not(target_arch = "wasm32"), async_trait)] #[auto_impl(&, Box, Arc)] pub trait Middleware: Sync + Send + Debug { type Error: Sync + Send + Error + FromErr<::Error>; type Provider: JsonRpcClient; type Inner: Middleware; /// The next middleware in the stack fn inner(&self) -> &Self::Inner; /// Convert a provider error into the associated error type by successively /// converting it to every intermediate middleware error fn convert_err(p: ProviderError) -> Self::Error { let e = ::Inner::convert_err(p); FromErr::from(e) } /// The HTTP or Websocket provider. fn provider(&self) -> &Provider { self.inner().provider() } fn default_sender(&self) -> Option
{ self.inner().default_sender() } async fn client_version(&self) -> Result { self.inner().client_version().await.map_err(FromErr::from) } /// Fill necessary details of a transaction for dispatch /// /// This function is defined on providers to behave as follows: /// 1. populate the `from` field with the default sender /// 2. resolve any ENS names in the tx `to` field /// 3. Estimate gas usage /// 4. Poll and set legacy or 1559 gas prices /// 5. Set the chain_id with the provider's, if not already set /// /// It does NOT set the nonce by default. /// /// Middleware are encouraged to override any values _before_ delegating /// to the inner implementation AND/OR modify the values provided by the /// default implementation _after_ delegating. /// /// E.g. a middleware wanting to double gas prices should consider doing so /// _after_ delegating and allowing the default implementation to poll gas. async fn fill_transaction( &self, tx: &mut TypedTransaction, block: Option, ) -> Result<(), Self::Error> { self.inner().fill_transaction(tx, block).await.map_err(FromErr::from) } async fn get_block_number(&self) -> Result { self.inner().get_block_number().await.map_err(FromErr::from) } async fn send_transaction + Send + Sync>( &self, tx: T, block: Option, ) -> Result, Self::Error> { self.inner().send_transaction(tx, block).await.map_err(FromErr::from) } /// Send a transaction with a simple escalation policy. /// /// `policy` should be a boxed function that maps `original_gas_price` /// and `number_of_previous_escalations` -> `new_gas_price`. /// /// e.g. `Box::new(|start, escalation_index| start * 1250.pow(escalations) / /// 1000.pow(escalations))` async fn send_escalating<'a>( &'a self, tx: &TypedTransaction, escalations: usize, policy: EscalationPolicy, ) -> Result, Self::Error> { let mut original = tx.clone(); self.fill_transaction(&mut original, None).await?; // set the nonce, if no nonce is found if original.nonce().is_none() { let nonce = self.get_transaction_count(tx.from().copied().unwrap_or_default(), None).await?; original.set_nonce(nonce); } let gas_price = original.gas_price().expect("filled"); let sign_futs: Vec<_> = (0..escalations) .map(|i| { let new_price = policy(gas_price, i); let mut r = original.clone(); r.set_gas_price(new_price); r }) .map(|req| async move { self.sign_transaction(&req, self.default_sender().unwrap_or_default()) .await .map(|sig| req.rlp_signed(&sig)) }) .collect(); // we reverse for convenience. Ensuring that we can always just // `pop()` the next tx off the back later let mut signed = join_all(sign_futs).await.into_iter().collect::, _>>()?; signed.reverse(); Ok(EscalatingPending::new(self.provider(), signed)) } async fn resolve_name(&self, ens_name: &str) -> Result { self.inner().resolve_name(ens_name).await.map_err(FromErr::from) } async fn lookup_address(&self, address: Address) -> Result { self.inner().lookup_address(address).await.map_err(FromErr::from) } async fn resolve_avatar(&self, ens_name: &str) -> Result { self.inner().resolve_avatar(ens_name).await.map_err(FromErr::from) } async fn resolve_nft(&self, token: erc::ERCNFT) -> Result { self.inner().resolve_nft(token).await.map_err(FromErr::from) } async fn resolve_field(&self, ens_name: &str, field: &str) -> Result { self.inner().resolve_field(ens_name, field).await.map_err(FromErr::from) } async fn get_block + Send + Sync>( &self, block_hash_or_number: T, ) -> Result>, Self::Error> { self.inner().get_block(block_hash_or_number).await.map_err(FromErr::from) } async fn get_block_with_txs + Send + Sync>( &self, block_hash_or_number: T, ) -> Result>, Self::Error> { self.inner().get_block_with_txs(block_hash_or_number).await.map_err(FromErr::from) } async fn get_uncle_count + Send + Sync>( &self, block_hash_or_number: T, ) -> Result { self.inner().get_uncle_count(block_hash_or_number).await.map_err(FromErr::from) } async fn get_uncle + Send + Sync>( &self, block_hash_or_number: T, idx: U64, ) -> Result>, Self::Error> { self.inner().get_uncle(block_hash_or_number, idx).await.map_err(FromErr::from) } async fn get_transaction_count + Send + Sync>( &self, from: T, block: Option, ) -> Result { self.inner().get_transaction_count(from, block).await.map_err(FromErr::from) } async fn estimate_gas( &self, tx: &TypedTransaction, block: Option, ) -> Result { self.inner().estimate_gas(tx, block).await.map_err(FromErr::from) } async fn call( &self, tx: &TypedTransaction, block: Option, ) -> Result { self.inner().call(tx, block).await.map_err(FromErr::from) } async fn syncing(&self) -> Result { self.inner().syncing().await.map_err(FromErr::from) } async fn get_chainid(&self) -> Result { self.inner().get_chainid().await.map_err(FromErr::from) } async fn get_net_version(&self) -> Result { self.inner().get_net_version().await.map_err(FromErr::from) } async fn get_balance + Send + Sync>( &self, from: T, block: Option, ) -> Result { self.inner().get_balance(from, block).await.map_err(FromErr::from) } async fn get_transaction>( &self, transaction_hash: T, ) -> Result, Self::Error> { self.inner().get_transaction(transaction_hash).await.map_err(FromErr::from) } async fn get_transaction_receipt>( &self, transaction_hash: T, ) -> Result, Self::Error> { self.inner().get_transaction_receipt(transaction_hash).await.map_err(FromErr::from) } async fn get_block_receipts + Send + Sync>( &self, block: T, ) -> Result, Self::Error> { self.inner().get_block_receipts(block).await.map_err(FromErr::from) } async fn get_gas_price(&self) -> Result { self.inner().get_gas_price().await.map_err(FromErr::from) } async fn estimate_eip1559_fees( &self, estimator: Option>) -> (U256, U256)>, ) -> Result<(U256, U256), Self::Error> { self.inner().estimate_eip1559_fees(estimator).await.map_err(FromErr::from) } async fn get_accounts(&self) -> Result, Self::Error> { self.inner().get_accounts().await.map_err(FromErr::from) } async fn send_raw_transaction<'a>( &'a self, tx: Bytes, ) -> Result, Self::Error> { self.inner().send_raw_transaction(tx).await.map_err(FromErr::from) } /// This returns true if either the middleware stack contains a `SignerMiddleware`, or the /// JSON-RPC provider has an unlocked key that can sign using the `eth_sign` call. If none of /// the above conditions are met, then the middleware stack is not capable of signing data. async fn is_signer(&self) -> bool { self.inner().is_signer().await } async fn sign + Send + Sync>( &self, data: T, from: &Address, ) -> Result { self.inner().sign(data, from).await.map_err(FromErr::from) } /// Sign a transaction via RPC call async fn sign_transaction( &self, tx: &TypedTransaction, from: Address, ) -> Result { self.inner().sign_transaction(tx, from).await.map_err(FromErr::from) } ////// Contract state async fn get_logs(&self, filter: &Filter) -> Result, Self::Error> { self.inner().get_logs(filter).await.map_err(FromErr::from) } /// Returns a stream of logs are loaded in pages of given page size fn get_logs_paginated<'a>( &'a self, filter: &Filter, page_size: u64, ) -> LogQuery<'a, Self::Provider> { self.inner().get_logs_paginated(filter, page_size) } async fn new_filter(&self, filter: FilterKind<'_>) -> Result { self.inner().new_filter(filter).await.map_err(FromErr::from) } async fn uninstall_filter + Send + Sync>( &self, id: T, ) -> Result { self.inner().uninstall_filter(id).await.map_err(FromErr::from) } async fn watch<'a>( &'a self, filter: &Filter, ) -> Result, Self::Error> { self.inner().watch(filter).await.map_err(FromErr::from) } async fn watch_pending_transactions( &self, ) -> Result, Self::Error> { self.inner().watch_pending_transactions().await.map_err(FromErr::from) } async fn get_filter_changes(&self, id: T) -> Result, Self::Error> where T: Into + Send + Sync, R: Serialize + DeserializeOwned + Send + Sync + Debug, { self.inner().get_filter_changes(id).await.map_err(FromErr::from) } async fn watch_blocks(&self) -> Result, Self::Error> { self.inner().watch_blocks().await.map_err(FromErr::from) } async fn get_code + Send + Sync>( &self, at: T, block: Option, ) -> Result { self.inner().get_code(at, block).await.map_err(FromErr::from) } async fn get_storage_at + Send + Sync>( &self, from: T, location: H256, block: Option, ) -> Result { self.inner().get_storage_at(from, location, block).await.map_err(FromErr::from) } async fn get_proof + Send + Sync>( &self, from: T, locations: Vec, block: Option, ) -> Result { self.inner().get_proof(from, locations, block).await.map_err(FromErr::from) } /// Returns an indication if this node is currently mining. async fn mining(&self) -> Result { self.inner().mining().await.map_err(FromErr::from) } // Personal namespace async fn import_raw_key( &self, private_key: Bytes, passphrase: String, ) -> Result { self.inner().import_raw_key(private_key, passphrase).await.map_err(FromErr::from) } async fn unlock_account + Send + Sync>( &self, account: T, passphrase: String, duration: Option, ) -> Result { self.inner().unlock_account(account, passphrase, duration).await.map_err(FromErr::from) } // Admin namespace async fn add_peer(&self, enode_url: String) -> Result { self.inner().add_peer(enode_url).await.map_err(FromErr::from) } async fn add_trusted_peer(&self, enode_url: String) -> Result { self.inner().add_trusted_peer(enode_url).await.map_err(FromErr::from) } async fn node_info(&self) -> Result { self.inner().node_info().await.map_err(FromErr::from) } async fn peers(&self) -> Result, Self::Error> { self.inner().peers().await.map_err(FromErr::from) } async fn remove_peer(&self, enode_url: String) -> Result { self.inner().remove_peer(enode_url).await.map_err(FromErr::from) } async fn remove_trusted_peer(&self, enode_url: String) -> Result { self.inner().remove_trusted_peer(enode_url).await.map_err(FromErr::from) } // Miner namespace /// Starts the miner with the given number of threads. If threads is nil, the number of workers /// started is equal to the number of logical CPUs that are usable by this process. If mining /// is already running, this method adjust the number of threads allowed to use and updates the /// minimum price required by the transaction pool. async fn start_mining(&self, threads: Option) -> Result<(), Self::Error> { self.inner().start_mining(threads).await.map_err(FromErr::from) } /// Stop terminates the miner, both at the consensus engine level as well as at /// the block creation level. async fn stop_mining(&self) -> Result<(), Self::Error> { self.inner().stop_mining().await.map_err(FromErr::from) } // Mempool inspection for Geth's API async fn txpool_content(&self) -> Result { self.inner().txpool_content().await.map_err(FromErr::from) } async fn txpool_inspect(&self) -> Result { self.inner().txpool_inspect().await.map_err(FromErr::from) } async fn txpool_status(&self) -> Result { self.inner().txpool_status().await.map_err(FromErr::from) } // Geth `trace` support /// After replaying any previous transactions in the same block, /// Replays a transaction, returning the traces configured with passed options async fn debug_trace_transaction( &self, tx_hash: TxHash, trace_options: GethDebugTracingOptions, ) -> Result { self.inner().debug_trace_transaction(tx_hash, trace_options).await.map_err(FromErr::from) } /// Executes the given call and returns a number of possible traces for it async fn debug_trace_call + Send + Sync>( &self, req: T, block: Option, trace_options: GethDebugTracingCallOptions, ) -> Result { self.inner().debug_trace_call(req, block, trace_options).await.map_err(FromErr::from) } // Parity `trace` support /// Executes the given call and returns a number of possible traces for it async fn trace_call + Send + Sync>( &self, req: T, trace_type: Vec, block: Option, ) -> Result { self.inner().trace_call(req, trace_type, block).await.map_err(FromErr::from) } async fn trace_call_many + Send + Sync>( &self, req: Vec<(T, Vec)>, block: Option, ) -> Result, Self::Error> { self.inner().trace_call_many(req, block).await.map_err(FromErr::from) } /// Traces a call to `eth_sendRawTransaction` without making the call, returning the traces async fn trace_raw_transaction( &self, data: Bytes, trace_type: Vec, ) -> Result { self.inner().trace_raw_transaction(data, trace_type).await.map_err(FromErr::from) } /// Replays a transaction, returning the traces async fn trace_replay_transaction( &self, hash: H256, trace_type: Vec, ) -> Result { self.inner().trace_replay_transaction(hash, trace_type).await.map_err(FromErr::from) } /// Replays all transactions in a block returning the requested traces for each transaction async fn trace_replay_block_transactions( &self, block: BlockNumber, trace_type: Vec, ) -> Result, Self::Error> { self.inner().trace_replay_block_transactions(block, trace_type).await.map_err(FromErr::from) } /// Returns traces created at given block async fn trace_block(&self, block: BlockNumber) -> Result, Self::Error> { self.inner().trace_block(block).await.map_err(FromErr::from) } /// Return traces matching the given filter async fn trace_filter(&self, filter: TraceFilter) -> Result, Self::Error> { self.inner().trace_filter(filter).await.map_err(FromErr::from) } /// Returns trace at the given position async fn trace_get + Send + Sync>( &self, hash: H256, index: Vec, ) -> Result { self.inner().trace_get(hash, index).await.map_err(FromErr::from) } /// Returns all traces of a given transaction async fn trace_transaction(&self, hash: H256) -> Result, Self::Error> { self.inner().trace_transaction(hash).await.map_err(FromErr::from) } // Parity namespace /// Returns all receipts for that block. Must be done on a parity node. async fn parity_block_receipts + Send + Sync>( &self, block: T, ) -> Result, Self::Error> { self.inner().parity_block_receipts(block).await.map_err(FromErr::from) } async fn subscribe( &self, params: T, ) -> Result, Self::Error> where T: Debug + Serialize + Send + Sync, R: DeserializeOwned + Send + Sync, ::Provider: PubsubClient, { self.inner().subscribe(params).await.map_err(FromErr::from) } async fn unsubscribe(&self, id: T) -> Result where T: Into + Send + Sync, ::Provider: PubsubClient, { self.inner().unsubscribe(id).await.map_err(FromErr::from) } async fn subscribe_blocks( &self, ) -> Result>, Self::Error> where ::Provider: PubsubClient, { self.inner().subscribe_blocks().await.map_err(FromErr::from) } async fn subscribe_pending_txs( &self, ) -> Result, Self::Error> where ::Provider: PubsubClient, { self.inner().subscribe_pending_txs().await.map_err(FromErr::from) } async fn subscribe_logs<'a>( &'a self, filter: &Filter, ) -> Result, Self::Error> where ::Provider: PubsubClient, { self.inner().subscribe_logs(filter).await.map_err(FromErr::from) } async fn fee_history + serde::Serialize + Send + Sync>( &self, block_count: T, last_block: BlockNumber, reward_percentiles: &[f64], ) -> Result { self.inner() .fee_history(block_count, last_block, reward_percentiles) .await .map_err(FromErr::from) } async fn create_access_list( &self, tx: &TypedTransaction, block: Option, ) -> Result { self.inner().create_access_list(tx, block).await.map_err(FromErr::from) } } #[cfg(feature = "celo")] #[cfg_attr(target_arch = "wasm32", async_trait(?Send))] #[cfg_attr(not(target_arch = "wasm32"), async_trait)] pub trait CeloMiddleware: Middleware { async fn get_validators_bls_public_keys + Send + Sync>( &self, block_id: T, ) -> Result, ProviderError> { self.provider().get_validators_bls_public_keys(block_id).await.map_err(FromErr::from) } } #[allow(deprecated)] pub use test_provider::{GOERLI, MAINNET, ROPSTEN, SEPOLIA}; /// Pre-instantiated Infura HTTP clients which rotate through multiple API keys /// to prevent rate limits pub mod test_provider { use super::*; use crate::Http; use once_cell::sync::Lazy; use std::{convert::TryFrom, iter::Cycle, slice::Iter, sync::Mutex}; // List of infura keys to rotate through so we don't get rate limited const INFURA_KEYS: &[&str] = &["15e8aaed6f894d63a0f6a0206c006cdd"]; pub static MAINNET: Lazy = Lazy::new(|| TestProvider::new(INFURA_KEYS, "mainnet")); pub static GOERLI: Lazy = Lazy::new(|| TestProvider::new(INFURA_KEYS, "goerli")); pub static SEPOLIA: Lazy = Lazy::new(|| TestProvider::new(INFURA_KEYS, "sepolia")); #[deprecated = "Ropsten testnet has been deprecated in favor of Goerli or Sepolia."] pub static ROPSTEN: Lazy = Lazy::new(|| TestProvider::new(INFURA_KEYS, "ropsten")); #[derive(Debug)] pub struct TestProvider { network: String, keys: Mutex>>, } impl TestProvider { pub fn new(keys: &'static [&'static str], network: impl Into) -> Self { Self { keys: keys.iter().cycle().into(), network: network.into() } } pub fn url(&self) -> String { let Self { network, keys } = self; let key = keys.lock().unwrap().next().unwrap(); format!("https://{network}.infura.io/v3/{key}") } pub fn provider(&self) -> Provider { Provider::try_from(self.url().as_str()).unwrap() } #[cfg(feature = "ws")] pub async fn ws(&self) -> Provider { let url = format!( "wss://{}.infura.io/ws/v3/{}", self.network, self.keys.lock().unwrap().next().unwrap() ); Provider::connect(url.as_str()).await.unwrap() } } }