reth_provider/bundle_state/
state_reverts.rs
1use alloy_primitives::{B256, U256};
2use revm_database::states::RevertToSlot;
3use std::iter::Peekable;
4
5#[expect(missing_debug_implementations)]
8pub struct StorageRevertsIter<R: Iterator, W: Iterator> {
9 reverts: Peekable<R>,
10 wiped: Peekable<W>,
11}
12
13impl<R, W> StorageRevertsIter<R, W>
14where
15 R: Iterator<Item = (B256, RevertToSlot)>,
16 W: Iterator<Item = (B256, U256)>,
17{
18 pub fn new(
20 reverts: impl IntoIterator<IntoIter = R>,
21 wiped: impl IntoIterator<IntoIter = W>,
22 ) -> Self {
23 Self { reverts: reverts.into_iter().peekable(), wiped: wiped.into_iter().peekable() }
24 }
25
26 fn next_revert(&mut self) -> Option<(B256, U256)> {
28 self.reverts.next().map(|(key, revert)| (key, revert.to_previous_value()))
29 }
30
31 fn next_wiped(&mut self) -> Option<(B256, U256)> {
33 self.wiped.next()
34 }
35}
36
37impl<R, W> Iterator for StorageRevertsIter<R, W>
38where
39 R: Iterator<Item = (B256, RevertToSlot)>,
40 W: Iterator<Item = (B256, U256)>,
41{
42 type Item = (B256, U256);
43
44 fn next(&mut self) -> Option<Self::Item> {
47 match (self.reverts.peek(), self.wiped.peek()) {
48 (Some(revert), Some(wiped)) => {
49 use std::cmp::Ordering;
51 match revert.0.cmp(&wiped.0) {
52 Ordering::Less => self.next_revert(),
53 Ordering::Greater => self.next_wiped(),
54 Ordering::Equal => {
55 let (key, revert_to) = *revert;
57
58 let value = match revert_to {
59 RevertToSlot::Some(value) => value,
61 RevertToSlot::Destroyed => wiped.1,
63 };
64
65 self.next_revert();
67 self.next_wiped();
68
69 Some((key, value))
70 }
71 }
72 }
73 (Some(_revert), None) => self.next_revert(),
74 (None, Some(_wiped)) => self.next_wiped(),
75 (None, None) => None,
76 }
77 }
78}
79
80#[cfg(test)]
81mod tests {
82 use super::*;
83
84 #[test]
85 fn test_storage_reverts_iter_empty() {
86 let reverts: Vec<(B256, RevertToSlot)> = vec![];
88 let wiped: Vec<(B256, U256)> = vec![];
89
90 let iter = StorageRevertsIter::new(reverts, wiped);
92
93 let results: Vec<_> = iter.collect();
95
96 assert_eq!(results, vec![]);
98 }
99
100 #[test]
101 fn test_storage_reverts_iter_reverts_only() {
102 let reverts = vec![
104 (B256::from_slice(&[4; 32]), RevertToSlot::Destroyed),
105 (B256::from_slice(&[5; 32]), RevertToSlot::Some(U256::from(40))),
106 ];
107
108 let iter = StorageRevertsIter::new(reverts, vec![]);
110
111 let results: Vec<_> = iter.collect();
113
114 assert_eq!(
116 results,
117 vec![
118 (B256::from_slice(&[4; 32]), U256::ZERO), (B256::from_slice(&[5; 32]), U256::from(40)), ]
121 );
122 }
123
124 #[test]
125 fn test_storage_reverts_iter_wiped_only() {
126 let wiped = vec![
128 (B256::from_slice(&[6; 32]), U256::from(50)),
129 (B256::from_slice(&[7; 32]), U256::from(60)),
130 ];
131
132 let iter = StorageRevertsIter::new(vec![], wiped);
134
135 let results: Vec<_> = iter.collect();
137
138 assert_eq!(
140 results,
141 vec![
142 (B256::from_slice(&[6; 32]), U256::from(50)), (B256::from_slice(&[7; 32]), U256::from(60)), ]
145 );
146 }
147
148 #[test]
149 fn test_storage_reverts_iter_interleaved() {
150 let reverts = vec![
152 (B256::from_slice(&[8; 32]), RevertToSlot::Some(U256::from(70))),
153 (B256::from_slice(&[9; 32]), RevertToSlot::Some(U256::from(80))),
154 (B256::from_slice(&[15; 32]), RevertToSlot::Some(U256::from(90))),
156 ];
157
158 let wiped = vec![
159 (B256::from_slice(&[8; 32]), U256::from(75)), (B256::from_slice(&[10; 32]), U256::from(85)), ];
162
163 let iter = StorageRevertsIter::new(reverts, wiped);
165
166 let results: Vec<_> = iter.collect();
168
169 assert_eq!(
171 results,
172 vec![
173 (B256::from_slice(&[8; 32]), U256::from(70)), (B256::from_slice(&[9; 32]), U256::from(80)), (B256::from_slice(&[10; 32]), U256::from(85)), (B256::from_slice(&[15; 32]), U256::from(90)), ]
178 );
179 }
180}