Skip to main content

foundry_evm_hardforks/
lib.rs

1//! EVM hardfork definitions for Foundry.
2//!
3//! Provides [`FoundryHardfork`], a unified enum over Ethereum, Optimism, and Tempo hardforks
4//! with `FromStr`/`Serialize`/`Deserialize` support for CLI and config usage.
5
6use std::{
7    str::FromStr,
8    time::{SystemTime, UNIX_EPOCH},
9};
10
11use alloy_chains::Chain;
12use alloy_rpc_types::BlockNumberOrTag;
13use foundry_compilers::artifacts::EvmVersion;
14#[cfg(feature = "optimism")]
15use op_revm::OpSpecId;
16use revm::primitives::hardfork::SpecId;
17use serde::{Deserialize, Serialize};
18
19pub use alloy_hardforks::EthereumHardfork;
20#[cfg(feature = "optimism")]
21pub use alloy_op_hardforks::OpHardfork;
22pub use tempo_chainspec::hardfork::TempoHardfork;
23
24#[derive(Clone, Copy, Debug, PartialEq, Eq, PartialOrd, Ord, Serialize)]
25#[serde(into = "String")]
26pub enum FoundryHardfork {
27    Ethereum(EthereumHardfork),
28    #[cfg(feature = "optimism")]
29    Optimism(OpHardfork),
30    Tempo(TempoHardfork),
31}
32
33impl From<FoundryHardfork> for String {
34    fn from(fork: FoundryHardfork) -> Self {
35        match fork {
36            FoundryHardfork::Ethereum(h) => format!("{h}"),
37            #[cfg(feature = "optimism")]
38            FoundryHardfork::Optimism(h) => format!("optimism:{h}"),
39            FoundryHardfork::Tempo(h) => format!("tempo:{h}"),
40        }
41    }
42}
43
44impl<'de> Deserialize<'de> for FoundryHardfork {
45    fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
46    where
47        D: serde::Deserializer<'de>,
48    {
49        let s = String::deserialize(deserializer)?;
50        Self::from_str(&s).map_err(serde::de::Error::custom)
51    }
52}
53
54impl FromStr for FoundryHardfork {
55    type Err = String;
56
57    fn from_str(s: &str) -> Result<Self, Self::Err> {
58        let raw = s.trim();
59
60        let Some((ns, fork_raw)) = raw.split_once(':') else {
61            return EthereumHardfork::from_str(raw)
62                .map(Self::Ethereum)
63                .map_err(|_| format!("unknown ethereum hardfork '{raw}'"));
64        };
65
66        let ns = ns.trim().to_ascii_lowercase();
67        let fork = fork_raw.trim().to_ascii_lowercase().replace(['-', ' '], "_");
68
69        match ns.as_str() {
70            "eth" | "ethereum" => EthereumHardfork::from_str(&fork)
71                .map(Self::Ethereum)
72                .map_err(|_| format!("unknown ethereum hardfork '{fork_raw}'")),
73
74            #[cfg(feature = "optimism")]
75            "op" | "optimism" => OpHardfork::from_str(&fork)
76                .map(Self::Optimism)
77                .map_err(|_| format!("unknown optimism hardfork '{fork_raw}'")),
78
79            "t" | "tempo" => TempoHardfork::from_str(&fork)
80                .map(Self::Tempo)
81                .map_err(|_| format!("unknown tempo hardfork '{fork_raw}'")),
82            _ => EthereumHardfork::from_str(&fork)
83                .map(Self::Ethereum)
84                .map_err(|_| format!("unknown hardfork '{raw}'")),
85        }
86    }
87}
88
89impl FoundryHardfork {
90    pub const fn ethereum(h: EthereumHardfork) -> Self {
91        Self::Ethereum(h)
92    }
93
94    #[cfg(feature = "optimism")]
95    pub const fn optimism(h: OpHardfork) -> Self {
96        Self::Optimism(h)
97    }
98
99    pub const fn tempo(h: TempoHardfork) -> Self {
100        Self::Tempo(h)
101    }
102
103    /// Returns the hardfork name without a network namespace prefix.
104    pub fn name(&self) -> String {
105        match self {
106            Self::Ethereum(h) => format!("{h}"),
107            #[cfg(feature = "optimism")]
108            Self::Optimism(h) => format!("{h}"),
109            Self::Tempo(h) => format!("{h}"),
110        }
111    }
112
113    /// Returns the network namespace for this hardfork, or `None` for plain Ethereum.
114    ///
115    /// Mirrors the namespace prefix used in the `"network:hardfork"` serialization format.
116    pub const fn namespace(&self) -> Option<&'static str> {
117        match self {
118            Self::Ethereum(_) => None,
119            #[cfg(feature = "optimism")]
120            Self::Optimism(_) => Some("optimism"),
121            Self::Tempo(_) => Some("tempo"),
122        }
123    }
124
125    /// Auto-detect the active hardfork for a given chain at a specific timestamp.
126    ///
127    /// Tries Ethereum, then Optimism. Returns `None` for unknown chains.
128    pub fn from_chain_and_timestamp(chain_id: u64, timestamp: u64) -> Option<Self> {
129        let chain = Chain::from_id(chain_id);
130        if let Some(fork) = EthereumHardfork::from_chain_and_timestamp(chain, timestamp) {
131            return Some(Self::Ethereum(fork));
132        }
133        #[cfg(feature = "optimism")]
134        if let Some(fork) = OpHardfork::from_chain_and_timestamp(chain, timestamp) {
135            return Some(Self::Optimism(fork));
136        }
137        TempoHardfork::from_chain_and_timestamp(chain_id, timestamp).map(Self::Tempo)
138    }
139}
140
141impl From<EthereumHardfork> for FoundryHardfork {
142    fn from(value: EthereumHardfork) -> Self {
143        Self::Ethereum(value)
144    }
145}
146
147impl From<FoundryHardfork> for EthereumHardfork {
148    fn from(fork: FoundryHardfork) -> Self {
149        match fork {
150            FoundryHardfork::Ethereum(hardfork) => hardfork,
151            _ => Self::default(),
152        }
153    }
154}
155
156#[cfg(feature = "optimism")]
157impl From<OpHardfork> for FoundryHardfork {
158    fn from(value: OpHardfork) -> Self {
159        Self::Optimism(value)
160    }
161}
162
163#[cfg(feature = "optimism")]
164impl From<FoundryHardfork> for OpHardfork {
165    fn from(fork: FoundryHardfork) -> Self {
166        match fork {
167            FoundryHardfork::Optimism(hardfork) => hardfork,
168            _ => Self::default(),
169        }
170    }
171}
172
173impl From<TempoHardfork> for FoundryHardfork {
174    fn from(value: TempoHardfork) -> Self {
175        Self::Tempo(value)
176    }
177}
178
179impl From<FoundryHardfork> for TempoHardfork {
180    fn from(fork: FoundryHardfork) -> Self {
181        match fork {
182            FoundryHardfork::Tempo(hardfork) => hardfork,
183            _ => Self::default(),
184        }
185    }
186}
187
188impl From<FoundryHardfork> for SpecId {
189    fn from(fork: FoundryHardfork) -> Self {
190        match fork {
191            FoundryHardfork::Ethereum(hardfork) => spec_id_from_ethereum_hardfork(hardfork),
192            #[cfg(feature = "optimism")]
193            FoundryHardfork::Optimism(hardfork) => spec_id_from_optimism_hardfork(hardfork).into(),
194            FoundryHardfork::Tempo(hardfork) => hardfork.into(),
195        }
196    }
197}
198
199#[cfg(feature = "optimism")]
200impl From<FoundryHardfork> for OpSpecId {
201    fn from(fork: FoundryHardfork) -> Self {
202        match fork {
203            FoundryHardfork::Optimism(hardfork) => spec_id_from_optimism_hardfork(hardfork),
204            _ => Self::default(),
205        }
206    }
207}
208
209/// Map an `EthereumHardfork` enum into its corresponding `SpecId`.
210pub fn spec_id_from_ethereum_hardfork(hardfork: EthereumHardfork) -> SpecId {
211    match hardfork {
212        EthereumHardfork::Frontier => SpecId::FRONTIER,
213        EthereumHardfork::Homestead => SpecId::HOMESTEAD,
214        EthereumHardfork::Dao => SpecId::DAO_FORK,
215        EthereumHardfork::Tangerine => SpecId::TANGERINE,
216        EthereumHardfork::SpuriousDragon => SpecId::SPURIOUS_DRAGON,
217        EthereumHardfork::Byzantium => SpecId::BYZANTIUM,
218        EthereumHardfork::Constantinople => SpecId::CONSTANTINOPLE,
219        EthereumHardfork::Petersburg => SpecId::PETERSBURG,
220        EthereumHardfork::Istanbul => SpecId::ISTANBUL,
221        EthereumHardfork::MuirGlacier => SpecId::MUIR_GLACIER,
222        EthereumHardfork::Berlin => SpecId::BERLIN,
223        EthereumHardfork::London => SpecId::LONDON,
224        EthereumHardfork::ArrowGlacier => SpecId::ARROW_GLACIER,
225        EthereumHardfork::GrayGlacier => SpecId::GRAY_GLACIER,
226        EthereumHardfork::Paris => SpecId::MERGE,
227        EthereumHardfork::Shanghai => SpecId::SHANGHAI,
228        EthereumHardfork::Cancun => SpecId::CANCUN,
229        EthereumHardfork::Prague => SpecId::PRAGUE,
230        EthereumHardfork::Osaka => SpecId::OSAKA,
231        EthereumHardfork::Bpo1 | EthereumHardfork::Bpo2 => SpecId::OSAKA,
232        EthereumHardfork::Bpo3 | EthereumHardfork::Bpo4 | EthereumHardfork::Bpo5 => {
233            unimplemented!()
234        }
235        EthereumHardfork::Amsterdam => SpecId::AMSTERDAM,
236        f => unreachable!("unimplemented {}", f),
237    }
238}
239
240/// Map an `OptimismHardfork` enum into its corresponding `OpSpecId`.
241#[cfg(feature = "optimism")]
242pub fn spec_id_from_optimism_hardfork(hardfork: OpHardfork) -> OpSpecId {
243    match hardfork {
244        OpHardfork::Bedrock => OpSpecId::BEDROCK,
245        OpHardfork::Regolith => OpSpecId::REGOLITH,
246        OpHardfork::Canyon => OpSpecId::CANYON,
247        OpHardfork::Ecotone => OpSpecId::ECOTONE,
248        OpHardfork::Fjord => OpSpecId::FJORD,
249        OpHardfork::Granite => OpSpecId::GRANITE,
250        OpHardfork::Holocene => OpSpecId::HOLOCENE,
251        OpHardfork::Isthmus => OpSpecId::ISTHMUS,
252        OpHardfork::Jovian => OpSpecId::JOVIAN,
253        OpHardfork::Karst => OpSpecId::KARST,
254        OpHardfork::Interop => OpSpecId::INTEROP,
255        f => unreachable!("unimplemented {}", f),
256    }
257}
258
259/// Trait for converting an [`EvmVersion`] into a network-specific spec type.
260pub trait FromEvmVersion: From<FoundryHardfork> {
261    fn from_evm_version(version: EvmVersion) -> Self;
262}
263
264/// Trait for parsing and displaying a network-specific execution spec.
265pub trait ExecutionSpec: FromEvmVersion {
266    // Returns the user-facing name for the active execution spec.
267    fn evm_version_name(&self) -> String;
268
269    // Parses an unnamespaced hardfork name for the active network.
270    fn from_network_hardfork(_: &str) -> Option<Self> {
271        None
272    }
273
274    // Converts a namespaced Foundry hardfork if it belongs to this spec family.
275    fn from_foundry_hardfork(hardfork: FoundryHardfork) -> Option<Self>;
276}
277
278impl FromEvmVersion for SpecId {
279    fn from_evm_version(version: EvmVersion) -> Self {
280        match version {
281            EvmVersion::Homestead => Self::HOMESTEAD,
282            EvmVersion::TangerineWhistle => Self::TANGERINE,
283            EvmVersion::SpuriousDragon => Self::SPURIOUS_DRAGON,
284            EvmVersion::Byzantium => Self::BYZANTIUM,
285            EvmVersion::Constantinople => Self::CONSTANTINOPLE,
286            EvmVersion::Petersburg => Self::PETERSBURG,
287            EvmVersion::Istanbul => Self::ISTANBUL,
288            EvmVersion::Berlin => Self::BERLIN,
289            EvmVersion::London => Self::LONDON,
290            EvmVersion::Paris => Self::MERGE,
291            EvmVersion::Shanghai => Self::SHANGHAI,
292            EvmVersion::Cancun => Self::CANCUN,
293            EvmVersion::Prague => Self::PRAGUE,
294            EvmVersion::Osaka => Self::OSAKA,
295            EvmVersion::Amsterdam => Self::AMSTERDAM,
296        }
297    }
298}
299
300impl ExecutionSpec for SpecId {
301    // Returns the user-facing name for the active execution spec.
302    fn evm_version_name(&self) -> String {
303        self.to_string()
304    }
305
306    // Parses an unnamespaced Ethereum hardfork name.
307    fn from_network_hardfork(hardfork: &str) -> Option<Self> {
308        EthereumHardfork::from_str(hardfork).ok().map(spec_id_from_ethereum_hardfork)
309    }
310
311    // Converts only Ethereum namespaced hardforks to an Ethereum spec.
312    fn from_foundry_hardfork(hardfork: FoundryHardfork) -> Option<Self> {
313        match hardfork {
314            FoundryHardfork::Ethereum(hardfork) => Some(spec_id_from_ethereum_hardfork(hardfork)),
315            _ => None,
316        }
317    }
318}
319
320#[cfg(feature = "optimism")]
321impl FromEvmVersion for OpSpecId {
322    fn from_evm_version(version: EvmVersion) -> Self {
323        match version {
324            EvmVersion::Homestead
325            | EvmVersion::TangerineWhistle
326            | EvmVersion::SpuriousDragon
327            | EvmVersion::Byzantium
328            | EvmVersion::Constantinople
329            | EvmVersion::Petersburg
330            | EvmVersion::Istanbul
331            | EvmVersion::Berlin
332            | EvmVersion::London
333            | EvmVersion::Paris => Self::BEDROCK,
334            EvmVersion::Shanghai => Self::CANYON,
335            EvmVersion::Cancun => Self::ECOTONE,
336            EvmVersion::Prague => Self::ISTHMUS,
337            EvmVersion::Osaka | EvmVersion::Amsterdam => Self::KARST,
338        }
339    }
340}
341
342#[cfg(feature = "optimism")]
343impl ExecutionSpec for OpSpecId {
344    // Returns the user-facing name for the active execution spec.
345    fn evm_version_name(&self) -> String {
346        let name: &'static str = (*self).into();
347        name.to_string()
348    }
349
350    // Parses an unnamespaced Optimism hardfork name.
351    fn from_network_hardfork(hardfork: &str) -> Option<Self> {
352        OpHardfork::from_str(hardfork).ok().map(spec_id_from_optimism_hardfork)
353    }
354
355    // Converts only Optimism namespaced hardforks to an Optimism spec.
356    fn from_foundry_hardfork(hardfork: FoundryHardfork) -> Option<Self> {
357        match hardfork {
358            FoundryHardfork::Optimism(hardfork) => Some(spec_id_from_optimism_hardfork(hardfork)),
359            _ => None,
360        }
361    }
362}
363
364impl FromEvmVersion for TempoHardfork {
365    fn from_evm_version(_: EvmVersion) -> Self {
366        latest_active_tempo_hardfork()
367    }
368}
369
370impl ExecutionSpec for TempoHardfork {
371    // Returns the user-facing name for the active execution spec.
372    fn evm_version_name(&self) -> String {
373        self.to_string()
374    }
375
376    // Parses an unnamespaced Tempo hardfork name.
377    fn from_network_hardfork(hardfork: &str) -> Option<Self> {
378        Self::from_str(hardfork).ok()
379    }
380
381    // Converts only Tempo namespaced hardforks to a Tempo spec.
382    fn from_foundry_hardfork(hardfork: FoundryHardfork) -> Option<Self> {
383        match hardfork {
384            FoundryHardfork::Tempo(hardfork) => Some(hardfork),
385            _ => None,
386        }
387    }
388}
389
390/// Returns the spec id derived from [`EvmVersion`] for a given spec type.
391pub fn evm_spec_id<SPEC: FromEvmVersion>(evm_version: EvmVersion) -> SPEC {
392    SPEC::from_evm_version(evm_version)
393}
394
395/// Returns the latest Tempo hardfork that has an activation on a known Tempo network.
396pub fn latest_active_tempo_hardfork() -> TempoHardfork {
397    // Tempo currently publishes activation timestamps through chain-aware hardfork resolution.
398    let now = SystemTime::now()
399        .duration_since(UNIX_EPOCH)
400        .map(|duration| duration.as_secs())
401        .unwrap_or(u64::MAX);
402    TempoHardfork::from_chain_and_timestamp(4217, now)
403        .or_else(|| TempoHardfork::from_chain_and_timestamp(42431, now))
404        .unwrap_or_default()
405}
406
407// Parses an EVM version or network-specific hardfork into the given spec type.
408pub fn evm_spec_id_from_str<SPEC: ExecutionSpec>(evm_version: &str) -> Option<SPEC> {
409    let evm_version = evm_version.trim();
410
411    if let Ok(version) = EvmVersion::from_str(evm_version) {
412        return Some(evm_spec_id(version));
413    }
414
415    if let Some(spec) = SPEC::from_network_hardfork(evm_version) {
416        return Some(spec);
417    }
418
419    FoundryHardfork::from_str(evm_version).ok().and_then(SPEC::from_foundry_hardfork)
420}
421
422/// Convert a `BlockNumberOrTag` into an `EthereumHardfork`.
423pub fn ethereum_hardfork_from_block_tag(block: impl Into<BlockNumberOrTag>) -> EthereumHardfork {
424    let num = match block.into() {
425        BlockNumberOrTag::Earliest => 0,
426        BlockNumberOrTag::Number(num) => num,
427        _ => u64::MAX,
428    };
429
430    EthereumHardfork::from_mainnet_block_number(num)
431}
432
433#[cfg(test)]
434mod tests {
435    use super::*;
436    use alloy_hardforks::ethereum::mainnet::*;
437
438    #[test]
439    fn test_ethereum_spec_id_mapping() {
440        assert_eq!(spec_id_from_ethereum_hardfork(EthereumHardfork::Frontier), SpecId::FRONTIER);
441        assert_eq!(spec_id_from_ethereum_hardfork(EthereumHardfork::Homestead), SpecId::HOMESTEAD);
442
443        // Test latest hardforks
444        assert_eq!(spec_id_from_ethereum_hardfork(EthereumHardfork::Cancun), SpecId::CANCUN);
445        assert_eq!(spec_id_from_ethereum_hardfork(EthereumHardfork::Prague), SpecId::PRAGUE);
446        assert_eq!(spec_id_from_ethereum_hardfork(EthereumHardfork::Osaka), SpecId::OSAKA);
447        assert_eq!(spec_id_from_ethereum_hardfork(EthereumHardfork::Amsterdam), SpecId::AMSTERDAM);
448    }
449
450    #[test]
451    fn test_tempo_spec_id_mapping() {
452        assert_eq!(SpecId::from(TempoHardfork::Genesis), SpecId::OSAKA);
453    }
454
455    #[test]
456    fn test_tempo_evm_version_defaults_to_latest_active_hardfork() {
457        assert_eq!(latest_active_tempo_hardfork(), TempoHardfork::T4);
458        assert_eq!(evm_spec_id::<TempoHardfork>(EvmVersion::Osaka), TempoHardfork::T4);
459    }
460
461    #[test]
462    fn test_tempo_hardfork_from_chain_and_timestamp() {
463        assert_eq!(
464            FoundryHardfork::from_chain_and_timestamp(4217, u64::MAX),
465            Some(FoundryHardfork::Tempo(TempoHardfork::T4))
466        );
467    }
468
469    #[test]
470    fn test_evm_spec_id_from_str_parses_network_hardforks() {
471        assert_eq!(evm_spec_id_from_str::<TempoHardfork>("T3"), Some(TempoHardfork::T3));
472        assert_eq!(evm_spec_id_from_str::<TempoHardfork>("tempo:T2"), Some(TempoHardfork::T2));
473        assert_eq!(evm_spec_id_from_str::<TempoHardfork>("ethereum:prague"), None);
474    }
475
476    #[test]
477    fn test_hardfork_from_block_tag_numbers() {
478        assert_eq!(
479            ethereum_hardfork_from_block_tag(MAINNET_HOMESTEAD_BLOCK - 1),
480            EthereumHardfork::Frontier
481        );
482        assert_eq!(
483            ethereum_hardfork_from_block_tag(MAINNET_LONDON_BLOCK + 1),
484            EthereumHardfork::London
485        );
486    }
487
488    #[test]
489    fn test_from_chain_and_timestamp_ethereum_mainnet() {
490        assert_eq!(
491            FoundryHardfork::from_chain_and_timestamp(1, 0),
492            Some(FoundryHardfork::Ethereum(EthereumHardfork::Frontier))
493        );
494        // Shanghai activated at timestamp 1681338455 on mainnet
495        assert_eq!(
496            FoundryHardfork::from_chain_and_timestamp(1, 1_681_338_455),
497            Some(FoundryHardfork::Ethereum(EthereumHardfork::Shanghai))
498        );
499    }
500
501    #[test]
502    fn test_from_chain_and_timestamp_sepolia() {
503        let sepolia_chain_id = 11155111;
504        assert!(FoundryHardfork::from_chain_and_timestamp(sepolia_chain_id, u64::MAX).is_some());
505    }
506
507    #[test]
508    fn test_from_chain_and_timestamp_unknown_chain() {
509        assert_eq!(FoundryHardfork::from_chain_and_timestamp(999999, 0), None);
510    }
511
512    #[cfg(feature = "optimism")]
513    mod optimism {
514        use super::*;
515
516        #[test]
517        fn test_optimism_spec_id_mapping() {
518            assert_eq!(spec_id_from_optimism_hardfork(OpHardfork::Bedrock), OpSpecId::BEDROCK);
519            assert_eq!(spec_id_from_optimism_hardfork(OpHardfork::Regolith), OpSpecId::REGOLITH);
520
521            // Test latest hardforks
522            assert_eq!(spec_id_from_optimism_hardfork(OpHardfork::Holocene), OpSpecId::HOLOCENE);
523            assert_eq!(spec_id_from_optimism_hardfork(OpHardfork::Karst), OpSpecId::KARST);
524            assert_eq!(spec_id_from_optimism_hardfork(OpHardfork::Interop), OpSpecId::INTEROP);
525            assert_eq!(evm_spec_id::<OpSpecId>(EvmVersion::Osaka), OpSpecId::KARST);
526        }
527
528        #[test]
529        fn test_from_chain_and_timestamp_op_mainnet() {
530            let op_chain_id = 10;
531            assert!(matches!(
532                FoundryHardfork::from_chain_and_timestamp(op_chain_id, u64::MAX),
533                Some(FoundryHardfork::Optimism(_))
534            ));
535        }
536
537        #[test]
538        fn test_from_chain_and_timestamp_base() {
539            let base_chain_id = 8453;
540            assert!(matches!(
541                FoundryHardfork::from_chain_and_timestamp(base_chain_id, u64::MAX),
542                Some(FoundryHardfork::Optimism(_))
543            ));
544        }
545    }
546}