1use 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 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 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 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
209pub 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#[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
259pub trait FromEvmVersion: From<FoundryHardfork> {
261 fn from_evm_version(version: EvmVersion) -> Self;
262}
263
264pub trait ExecutionSpec: FromEvmVersion {
266 fn evm_version_name(&self) -> String;
268
269 fn from_network_hardfork(_: &str) -> Option<Self> {
271 None
272 }
273
274 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 fn evm_version_name(&self) -> String {
303 self.to_string()
304 }
305
306 fn from_network_hardfork(hardfork: &str) -> Option<Self> {
308 EthereumHardfork::from_str(hardfork).ok().map(spec_id_from_ethereum_hardfork)
309 }
310
311 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 fn evm_version_name(&self) -> String {
346 let name: &'static str = (*self).into();
347 name.to_string()
348 }
349
350 fn from_network_hardfork(hardfork: &str) -> Option<Self> {
352 OpHardfork::from_str(hardfork).ok().map(spec_id_from_optimism_hardfork)
353 }
354
355 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 fn evm_version_name(&self) -> String {
373 self.to_string()
374 }
375
376 fn from_network_hardfork(hardfork: &str) -> Option<Self> {
378 Self::from_str(hardfork).ok()
379 }
380
381 fn from_foundry_hardfork(hardfork: FoundryHardfork) -> Option<Self> {
383 match hardfork {
384 FoundryHardfork::Tempo(hardfork) => Some(hardfork),
385 _ => None,
386 }
387 }
388}
389
390pub fn evm_spec_id<SPEC: FromEvmVersion>(evm_version: EvmVersion) -> SPEC {
392 SPEC::from_evm_version(evm_version)
393}
394
395pub fn latest_active_tempo_hardfork() -> TempoHardfork {
397 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
407pub 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
422pub 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 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 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 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}