use crate::{Provider, ProviderLayer, RootProvider};
use alloy_network::{Ethereum, Network};
use alloy_node_bindings::{Anvil, AnvilInstance};
use reqwest::Url;
use std::{
marker::PhantomData,
sync::{Arc, OnceLock},
};
#[derive(Debug, Clone, Default)]
pub struct AnvilLayer {
anvil: Anvil,
instance: OnceLock<Arc<AnvilInstance>>,
}
impl AnvilLayer {
pub fn instance(&self) -> &Arc<AnvilInstance> {
self.instance.get_or_init(|| Arc::new(self.anvil.clone().spawn()))
}
#[doc(alias = "http_endpoint_url")]
pub fn endpoint_url(&self) -> Url {
self.instance().endpoint_url()
}
pub fn ws_endpoint_url(&self) -> Url {
self.instance().ws_endpoint_url()
}
}
impl From<Anvil> for AnvilLayer {
fn from(anvil: Anvil) -> Self {
Self { anvil, instance: OnceLock::new() }
}
}
impl<P: Provider<N>, N: Network> ProviderLayer<P, N> for AnvilLayer {
type Provider = AnvilProvider<P, N>;
fn layer(&self, inner: P) -> Self::Provider {
let anvil = self.instance();
AnvilProvider::new(inner, anvil.clone())
}
}
#[derive(Clone, Debug)]
pub struct AnvilProvider<P, N = Ethereum> {
inner: P,
anvil: Arc<AnvilInstance>,
_marker: PhantomData<N>,
}
impl<P: Provider<N>, N: Network> AnvilProvider<P, N> {
#[expect(clippy::missing_const_for_fn)]
pub fn new(inner: P, anvil: Arc<AnvilInstance>) -> Self {
Self { inner, anvil, _marker: PhantomData }
}
pub const fn anvil(&self) -> &Arc<AnvilInstance> {
&self.anvil
}
}
impl<P: Provider<N>, N: Network> Provider<N> for AnvilProvider<P, N> {
#[inline(always)]
fn root(&self) -> &RootProvider<N> {
self.inner.root()
}
}