@@ -3,11 +3,11 @@ use alloy::{
3
3
} ;
4
4
use anyhow:: { anyhow, Context as AnyhowContext , Result } ;
5
5
6
+ use crate :: clients:: beacon:: types:: BlockHeader ;
6
7
use tracing:: { debug, info} ;
7
8
8
9
use crate :: {
9
10
clients:: {
10
- beacon:: types:: BlockHeader ,
11
11
blobscan:: types:: { Blob , BlobscanBlock , Block , Transaction } ,
12
12
common:: ClientError ,
13
13
} ,
@@ -20,6 +20,23 @@ use self::helpers::{create_tx_hash_versioned_hashes_mapping, create_versioned_ha
20
20
pub mod error;
21
21
mod helpers;
22
22
23
+ pub struct BlockData {
24
+ pub root : B256 ,
25
+ pub parent_root : B256 ,
26
+ pub slot : u32 ,
27
+ pub execution_block_hash : B256 ,
28
+ }
29
+
30
+ impl From < & BlockData > for BlockHeader {
31
+ fn from ( block : & BlockData ) -> Self {
32
+ BlockHeader {
33
+ root : block. root ,
34
+ parent_root : block. parent_root ,
35
+ slot : block. slot ,
36
+ }
37
+ }
38
+ }
39
+
23
40
pub struct SlotsProcessor < T > {
24
41
context : Box < dyn CommonContext < T > > ,
25
42
pub last_processed_block : Option < BlockHeader > ,
@@ -68,7 +85,13 @@ impl SlotsProcessor<ReqwestTransport> {
68
85
if prev_block_header. root != block_header. parent_root {
69
86
self . process_reorg ( & prev_block_header, & block_header)
70
87
. await
71
- . map_err ( |err| SlotsProcessorError :: ReorgedFailure ( err) ) ?;
88
+ . map_err ( |error| SlotsProcessorError :: FailedReorgProcessing {
89
+ old_slot : prev_block_header. slot ,
90
+ new_slot : block_header. slot ,
91
+ new_head_block_root : block_header. root ,
92
+ old_head_block_root : prev_block_header. root ,
93
+ error,
94
+ } ) ?;
72
95
}
73
96
}
74
97
@@ -85,25 +108,40 @@ impl SlotsProcessor<ReqwestTransport> {
85
108
& mut self ,
86
109
slot : u32 ,
87
110
) -> Result < Option < BlockHeader > , SlotProcessingError > {
88
- let beacon_client = self . context . beacon_client ( ) ;
89
- let blobscan_client = self . context . blobscan_client ( ) ;
90
- let provider = self . context . provider ( ) ;
91
-
92
- let beacon_block_header = Some ( match beacon_client. get_block_header ( slot. into ( ) ) . await ? {
111
+ let beacon_block_header = match self
112
+ . context
113
+ . beacon_client ( )
114
+ . get_block_header ( slot. into ( ) )
115
+ . await ?
116
+ {
93
117
Some ( header) => header,
94
118
None => {
95
119
debug ! ( slot, "Skipping as there is no beacon block header" ) ;
96
120
97
121
return Ok ( None ) ;
98
122
}
99
- } ) ;
123
+ } ;
124
+
125
+ self . process_block ( & beacon_block_header) . await ?;
126
+
127
+ Ok ( Some ( beacon_block_header) )
128
+ }
129
+
130
+ async fn process_block (
131
+ & self ,
132
+ beacon_block_header : & BlockHeader ,
133
+ ) -> Result < ( ) , SlotProcessingError > {
134
+ let beacon_client = self . context . beacon_client ( ) ;
135
+ let blobscan_client = self . context . blobscan_client ( ) ;
136
+ let provider = self . context . provider ( ) ;
137
+ let slot = beacon_block_header. slot ;
100
138
101
139
let beacon_block = match beacon_client. get_block ( slot. into ( ) ) . await ? {
102
140
Some ( block) => block,
103
141
None => {
104
142
debug ! ( slot = slot, "Skipping as there is no beacon block" ) ;
105
143
106
- return Ok ( None ) ;
144
+ return Ok ( ( ) ) ;
107
145
}
108
146
} ;
109
147
@@ -115,7 +153,7 @@ impl SlotsProcessor<ReqwestTransport> {
115
153
"Skipping as beacon block doesn't contain execution payload"
116
154
) ;
117
155
118
- return Ok ( beacon_block_header ) ;
156
+ return Ok ( ( ) ) ;
119
157
}
120
158
} ;
121
159
@@ -130,7 +168,7 @@ impl SlotsProcessor<ReqwestTransport> {
130
168
"Skipping as beacon block doesn't contain blob kzg commitments"
131
169
) ;
132
170
133
- return Ok ( beacon_block_header ) ;
171
+ return Ok ( ( ) ) ;
134
172
}
135
173
136
174
let execution_block_hash = execution_payload. block_hash ;
@@ -160,15 +198,15 @@ impl SlotsProcessor<ReqwestTransport> {
160
198
if blobs. is_empty ( ) {
161
199
debug ! ( slot, "Skipping as blobs sidecar is empty" ) ;
162
200
163
- return Ok ( beacon_block_header ) ;
201
+ return Ok ( ( ) ) ;
164
202
} else {
165
203
blobs
166
204
}
167
205
}
168
206
None => {
169
207
debug ! ( slot, "Skipping as there is no blobs sidecar" ) ;
170
208
171
- return Ok ( beacon_block_header ) ;
209
+ return Ok ( ( ) ) ;
172
210
}
173
211
} ;
174
212
@@ -217,19 +255,22 @@ impl SlotsProcessor<ReqwestTransport> {
217
255
218
256
info ! ( slot, block_number, "Block indexed successfully" ) ;
219
257
220
- Ok ( beacon_block_header )
258
+ Ok ( ( ) )
221
259
}
222
260
261
+ /// Handles reorgs by rewinding the blobscan blocks to the common ancestor and forwarding to the new head.
223
262
async fn process_reorg (
224
263
& mut self ,
225
264
old_head_header : & BlockHeader ,
226
265
new_head_header : & BlockHeader ,
227
- ) -> Result < ( ) , ClientError > {
266
+ ) -> Result < ( ) , anyhow :: Error > {
228
267
let mut current_old_slot = old_head_header. slot ;
229
268
230
- let mut rewinded_execution_blocks : Vec < B256 > = vec ! [ ] ;
269
+ let mut rewinded_blocks : Vec < B256 > = vec ! [ ] ;
231
270
232
271
loop {
272
+ // We iterate over blocks by slot and not block root as blobscan blocks don't
273
+ // have parent root we can use to traverse the chain
233
274
let old_blobscan_block = match self
234
275
. context
235
276
. blobscan_client ( )
@@ -240,56 +281,77 @@ impl SlotsProcessor<ReqwestTransport> {
240
281
None => {
241
282
current_old_slot -= 1 ;
242
283
284
+ // TODO: use fork slot instead of 0 as a stop condition to avoid long loops
243
285
if current_old_slot == 0 {
244
- return Err ( anyhow ! (
245
- "No blobscan block found for old head slot {}" ,
246
- old_head_header. slot
247
- )
248
- . into ( ) ) ;
286
+ return Err ( anyhow ! ( "No common block found" ) . into ( ) ) ;
249
287
}
250
288
251
289
continue ;
252
290
}
253
291
} ;
254
292
255
- let forwarded_execution_blocks = self
256
- . get_canonical_execution_blocks ( new_head_header. root , & old_blobscan_block )
293
+ let canonical_block_path = self
294
+ . get_canonical_block_path ( & old_blobscan_block , new_head_header. root )
257
295
. await ?;
258
-
259
- rewinded_execution_blocks. push ( old_blobscan_block. hash ) ;
260
-
261
- if !forwarded_execution_blocks. is_empty ( ) {
262
- let rewinded_blocks_count = rewinded_execution_blocks. len ( ) ;
263
- let forwarded_blocks_count = forwarded_execution_blocks. len ( ) ;
296
+ let canonical_block_path = canonical_block_path. into_iter ( ) . rev ( ) . collect :: < Vec < _ > > ( ) ;
297
+
298
+ // If a path exists, we've found the common ancient block
299
+ // and can proceed with handling the reorg.
300
+ if !canonical_block_path. is_empty ( ) {
301
+ let rewinded_blocks_count = rewinded_blocks. len ( ) ;
302
+ let forwarded_blocks_count = canonical_block_path. len ( ) ;
303
+
304
+ let canonical_block_headers: Vec < BlockHeader > = canonical_block_path
305
+ . iter ( )
306
+ . map ( |block| block. into ( ) )
307
+ . collect :: < Vec < _ > > ( ) ;
308
+
309
+ // If the new canonical block path includes blocks beyond the new head block,
310
+ // they were skipped and must be processed.
311
+ for block in canonical_block_headers. iter ( ) {
312
+ if block. slot != new_head_header. slot {
313
+ self . process_block ( block)
314
+ . await
315
+ . with_context ( || format ! ( "Failed to sync forwarded block" ) ) ?;
316
+ }
317
+ }
264
318
265
319
info ! (
266
320
new_slot = new_head_header. slot,
267
321
old_slot = old_head_header. slot,
268
322
"Reorg detected! rewinded blocks: {rewinded_blocks_count}, forwarded blocks: {forwarded_blocks_count}" ,
269
323
) ;
324
+
325
+ let forwarded_blocks = canonical_block_path
326
+ . iter ( )
327
+ . map ( |block| block. execution_block_hash )
328
+ . collect :: < Vec < _ > > ( ) ;
329
+
270
330
self . context
271
331
. blobscan_client ( )
272
- . handle_reorg ( rewinded_execution_blocks , forwarded_execution_blocks )
332
+ . handle_reorg ( rewinded_blocks , forwarded_blocks )
273
333
. await ?;
274
334
275
335
return Ok ( ( ) ) ;
276
336
}
337
+
338
+ rewinded_blocks. push ( old_blobscan_block. hash ) ;
277
339
}
278
340
}
279
341
280
- async fn get_canonical_execution_blocks (
342
+ /// Returns the path of blocks with execution payload from the head block to the provided block.
343
+ async fn get_canonical_block_path (
281
344
& mut self ,
282
- canonical_block_root : B256 ,
283
345
blobscan_block : & BlobscanBlock ,
284
- ) -> Result < Vec < B256 > , ClientError > {
346
+ head_block_root : B256 ,
347
+ ) -> Result < Vec < BlockData > , ClientError > {
285
348
let beacon_client = self . context . beacon_client ( ) ;
286
- let mut canonical_execution_blocks: Vec < B256 > = vec ! [ ] ;
349
+ let mut canonical_execution_blocks: Vec < BlockData > = vec ! [ ] ;
287
350
288
- let mut canonical_block = match beacon_client. get_block ( canonical_block_root. into ( ) ) . await ?
289
- {
351
+ let mut canonical_block = match beacon_client. get_block ( head_block_root. into ( ) ) . await ? {
290
352
Some ( block) => block,
291
353
None => {
292
- return Ok ( canonical_execution_blocks ) ;
354
+ return Ok ( vec ! [ ] ) ;
293
355
}
294
356
} ;
295
357
@@ -299,28 +361,39 @@ impl SlotsProcessor<ReqwestTransport> {
299
361
}
300
362
}
301
363
364
+ let mut current_canonical_block_root = head_block_root;
365
+
302
366
while canonical_block. message . parent_root != B256 :: ZERO {
367
+ let canonical_block_parent_root = canonical_block. message . parent_root ;
368
+
303
369
if canonical_block. message . slot < blobscan_block. slot {
304
370
return Ok ( vec ! [ ] ) ;
305
371
}
306
372
307
- if let Some ( execution_payload) = canonical_block. message . body . execution_payload {
373
+ if let Some ( execution_payload) = & canonical_block. message . body . execution_payload {
308
374
if execution_payload. block_hash == blobscan_block. hash {
309
375
return Ok ( canonical_execution_blocks) ;
310
376
}
311
377
312
- canonical_execution_blocks. push ( execution_payload. block_hash ) ;
378
+ canonical_execution_blocks. push ( BlockData {
379
+ root : current_canonical_block_root,
380
+ parent_root : canonical_block_parent_root,
381
+ slot : canonical_block. message . slot ,
382
+ execution_block_hash : execution_payload. block_hash ,
383
+ } ) ;
313
384
}
314
385
315
386
canonical_block = match beacon_client
316
- . get_block ( canonical_block . message . parent_root . into ( ) )
387
+ . get_block ( canonical_block_parent_root . into ( ) )
317
388
. await ?
318
389
{
319
390
Some ( block) => block,
320
391
None => {
321
392
return Ok ( canonical_execution_blocks) ;
322
393
}
323
394
} ;
395
+
396
+ current_canonical_block_root = canonical_block_parent_root;
324
397
}
325
398
326
399
Ok ( vec ! [ ] )
0 commit comments