1
- use crate :: archiver:: { Archiver , Config , STARTUP_FETCH_BLOB_MAXIMUM_RETRIES } ;
2
- use again:: RetryPolicy ;
3
- use blob_archiver_beacon:: beacon_client:: BeaconClientEth2 ;
4
- use blob_archiver_beacon:: blob_test_helper;
5
- use blob_archiver_storage:: fs:: FSStorage ;
6
- use clap:: Parser ;
7
- use eth2:: types:: BlockId ;
8
- use eth2:: { BeaconNodeHttpClient , SensitiveUrl , Timeouts } ;
9
- use serde:: Serialize ;
10
1
use std:: fs;
11
2
use std:: path:: PathBuf ;
12
3
use std:: str:: FromStr ;
13
4
use std:: sync:: Arc ;
14
5
use std:: time:: Duration ;
6
+
7
+ use again:: RetryPolicy ;
8
+ use clap:: Parser ;
9
+ use eth2:: types:: { BlockId , Hash256 } ;
10
+ use eth2:: { BeaconNodeHttpClient , SensitiveUrl , Timeouts } ;
11
+ use serde:: { Deserialize , Serialize } ;
15
12
use tokio:: sync:: Mutex ;
16
13
use tracing:: log:: error;
17
14
use tracing_appender:: rolling:: { RollingFileAppender , Rotation } ;
18
15
use tracing_subscriber:: layer:: SubscriberExt ;
19
16
use tracing_subscriber:: util:: SubscriberInitExt ;
20
17
use tracing_subscriber:: { fmt, EnvFilter } ;
21
18
19
+ use blob_archiver_beacon:: beacon_client:: BeaconClientEth2 ;
20
+ use blob_archiver_beacon:: { beacon_client, blob_test_helper} ;
21
+ use blob_archiver_storage:: fs:: FSStorage ;
22
+ use blob_archiver_storage:: s3:: S3Config ;
23
+ use blob_archiver_storage:: storage;
24
+ use blob_archiver_storage:: storage:: StorageType ;
25
+
26
+ use crate :: archiver:: { Archiver , Config , STARTUP_FETCH_BLOB_MAXIMUM_RETRIES } ;
27
+
22
28
mod api;
23
29
mod archiver;
24
30
@@ -27,7 +33,9 @@ static INIT: std::sync::Once = std::sync::Once::new();
27
33
#[ tokio:: main]
28
34
async fn main ( ) {
29
35
let args = CliArgs :: parse ( ) ;
30
- println ! ( "{}" , args. verbose) ;
36
+
37
+ let config = args. to_config ( ) ;
38
+ println ! ( "{:#?}" , config) ;
31
39
init_logging ( 0 , None , None ) ;
32
40
let beacon_client = BeaconNodeHttpClient :: new (
33
41
SensitiveUrl :: from_str ( "https://ethereum-beacon-api.publicnode.com" ) . unwrap ( ) ,
@@ -42,6 +50,7 @@ async fn main() {
42
50
origin_block : * blob_test_helper:: ORIGIN_BLOCK ,
43
51
beacon_config : Default :: default ( ) ,
44
52
storage_config : Default :: default ( ) ,
53
+ log_config : Default :: default ( ) ,
45
54
} ;
46
55
let archiver = Archiver :: new (
47
56
Arc :: new ( Mutex :: new ( beacon_client_eth2) ) ,
@@ -128,14 +137,14 @@ pub fn setup_tracing(
128
137
129
138
#[ derive( Parser , Serialize ) ]
130
139
pub struct CliArgs {
131
- #[ clap( short= 'v' , long, action = clap:: ArgAction :: Count , default_value = "3" ) ]
140
+ #[ clap( short = 'v' , long, action = clap:: ArgAction :: Count , default_value = "3" ) ]
132
141
verbose : u8 ,
133
142
134
- #[ clap( long, default_value = "logs" ) ]
135
- log_dir : String ,
143
+ #[ clap( long) ]
144
+ log_dir : Option < String > ,
136
145
137
- #[ clap( long, default_value = "DAILY" ) ]
138
- log_rotation : String ,
146
+ #[ clap( long, help = "Log rotation values: DAILY, HOURLY, MINUTELY, NEVER " ) ]
147
+ log_rotation : Option < String > ,
139
148
140
149
#[ clap( long, required = true ) ]
141
150
beacon_endpoint : String ,
@@ -169,3 +178,243 @@ pub struct CliArgs {
169
178
#[ clap( long) ]
170
179
fs_dir : Option < String > ,
171
180
}
181
+
182
+ impl CliArgs {
183
+ fn to_config ( & self ) -> Config {
184
+ let log_config = LogConfig {
185
+ log_dir : self . log_dir . as_ref ( ) . map ( PathBuf :: from) ,
186
+ log_rotation : self . log_rotation . clone ( ) ,
187
+ verbose : self . verbose ,
188
+ } ;
189
+
190
+ let beacon_config = beacon_client:: Config {
191
+ beacon_endpoint : self . beacon_endpoint . clone ( ) ,
192
+ beacon_client_timeout : Duration :: from_secs ( self . beacon_client_timeout ) ,
193
+ } ;
194
+
195
+ let storage_type = StorageType :: from_str ( self . storage_type . as_str ( ) ) . unwrap ( ) ;
196
+
197
+ let s3_config = if storage_type == StorageType :: S3 {
198
+ Some ( S3Config {
199
+ endpoint : self . s3_endpoint . clone ( ) . unwrap ( ) ,
200
+ bucket : self . s3_bucket . clone ( ) . unwrap ( ) ,
201
+ path : self . s3_path . clone ( ) . unwrap ( ) ,
202
+ compression : self . s3_compress . unwrap ( ) ,
203
+ } )
204
+ } else {
205
+ None
206
+ } ;
207
+
208
+ let fs_dir = self . fs_dir . as_ref ( ) . map ( PathBuf :: from) ;
209
+
210
+ let storage_config = storage:: Config {
211
+ storage_type,
212
+ s3_config,
213
+ fs_dir,
214
+ } ;
215
+
216
+ let mut padded_hex = self
217
+ . origin_block
218
+ . strip_prefix ( "0x" )
219
+ . unwrap_or ( & self . origin_block )
220
+ . to_string ( ) ;
221
+ padded_hex = format ! ( "{:0<64}" , padded_hex) ;
222
+ let origin_block = Hash256 :: from_slice ( & hex:: decode ( padded_hex) . expect ( "Invalid hex" ) ) ;
223
+
224
+ Config {
225
+ poll_interval : Duration :: from_secs ( self . poll_interval ) ,
226
+ listen_addr : self . listen_addr . clone ( ) ,
227
+ origin_block,
228
+ beacon_config,
229
+ storage_config,
230
+ log_config,
231
+ }
232
+ }
233
+ }
234
+
235
+ #[ allow( dead_code) ]
236
+ fn to_rotation ( s : & str ) -> Rotation {
237
+ match s {
238
+ "DAILY" => Rotation :: DAILY ,
239
+ "HOURLY" => Rotation :: HOURLY ,
240
+ "MINUTELY" => Rotation :: MINUTELY ,
241
+ _ => Rotation :: NEVER ,
242
+ }
243
+ }
244
+
245
+ #[ cfg( test) ]
246
+ mod tests {
247
+ use std:: path:: PathBuf ;
248
+ use std:: time:: Duration ;
249
+
250
+ use eth2:: types:: Hash256 ;
251
+
252
+ use blob_archiver_storage:: s3:: S3Config ;
253
+
254
+ use crate :: storage:: StorageType ;
255
+ use crate :: CliArgs ;
256
+
257
+ #[ test]
258
+ fn test_cli_args_to_config_s3_storage ( ) {
259
+ let cli_args = CliArgs {
260
+ verbose : 3 ,
261
+ log_dir : Some ( "logs" . to_string ( ) ) ,
262
+ log_rotation : Some ( "DAILY" . to_string ( ) ) ,
263
+ beacon_endpoint : "http://localhost:5052" . to_string ( ) ,
264
+ beacon_client_timeout : 10 ,
265
+ poll_interval : 6 ,
266
+ listen_addr : "0.0.0.0:8000" . to_string ( ) ,
267
+ origin_block : "0x1234" . to_string ( ) ,
268
+ storage_type : "s3" . to_string ( ) ,
269
+ s3_endpoint : Some ( "https://s3.amazonaws.com" . to_string ( ) ) ,
270
+ s3_bucket : Some ( "my-bucket" . to_string ( ) ) ,
271
+ s3_path : Some ( "my-path" . to_string ( ) ) ,
272
+ s3_compress : Some ( true ) ,
273
+ fs_dir : None ,
274
+ } ;
275
+
276
+ let config = cli_args. to_config ( ) ;
277
+
278
+ assert_eq ! ( config. poll_interval, Duration :: from_secs( 6 ) ) ;
279
+ assert_eq ! ( config. listen_addr, "0.0.0.0:8000" ) ;
280
+ assert_eq ! (
281
+ config. origin_block,
282
+ Hash256 :: from_slice(
283
+ & hex:: decode( "1234000000000000000000000000000000000000000000000000000000000000" )
284
+ . unwrap( )
285
+ )
286
+ ) ;
287
+ assert_eq ! (
288
+ config. beacon_config. beacon_endpoint,
289
+ "http://localhost:5052"
290
+ ) ;
291
+ assert_eq ! (
292
+ config. beacon_config. beacon_client_timeout,
293
+ Duration :: from_secs( 10 )
294
+ ) ;
295
+ assert_eq ! ( config. storage_config. storage_type, StorageType :: S3 ) ;
296
+ assert_eq ! (
297
+ config. storage_config. s3_config,
298
+ Some ( S3Config {
299
+ endpoint: "https://s3.amazonaws.com" . to_string( ) ,
300
+ bucket: "my-bucket" . to_string( ) ,
301
+ path: "my-path" . to_string( ) ,
302
+ compression: true ,
303
+ } )
304
+ ) ;
305
+ assert_eq ! ( config. storage_config. fs_dir, None ) ;
306
+ }
307
+
308
+ #[ test]
309
+ fn test_cli_args_to_config_fs_storage ( ) {
310
+ let cli_args = CliArgs {
311
+ verbose : 3 ,
312
+ log_dir : Some ( "logs" . to_string ( ) ) ,
313
+ log_rotation : Some ( "DAILY" . to_string ( ) ) ,
314
+ beacon_endpoint : "http://localhost:5052" . to_string ( ) ,
315
+ beacon_client_timeout : 10 ,
316
+ poll_interval : 6 ,
317
+ listen_addr : "0.0.0.0:8000" . to_string ( ) ,
318
+ origin_block : "0xabcd" . to_string ( ) ,
319
+ storage_type : "fs" . to_string ( ) ,
320
+ s3_endpoint : None ,
321
+ s3_bucket : None ,
322
+ s3_path : None ,
323
+ s3_compress : None ,
324
+ fs_dir : Some ( "/path/to/storage" . to_string ( ) ) ,
325
+ } ;
326
+
327
+ let config = cli_args. to_config ( ) ;
328
+
329
+ assert_eq ! ( config. poll_interval, Duration :: from_secs( 6 ) ) ;
330
+ assert_eq ! ( config. listen_addr, "0.0.0.0:8000" ) ;
331
+ assert_eq ! (
332
+ config. origin_block,
333
+ Hash256 :: from_slice(
334
+ & hex:: decode( "abcd000000000000000000000000000000000000000000000000000000000000" )
335
+ . unwrap( )
336
+ )
337
+ ) ;
338
+ assert_eq ! (
339
+ config. beacon_config. beacon_endpoint,
340
+ "http://localhost:5052"
341
+ ) ;
342
+ assert_eq ! (
343
+ config. beacon_config. beacon_client_timeout,
344
+ Duration :: from_secs( 10 )
345
+ ) ;
346
+ assert_eq ! ( config. storage_config. storage_type, StorageType :: FS ) ;
347
+ assert_eq ! ( config. storage_config. s3_config, None ) ;
348
+ assert_eq ! (
349
+ config. storage_config. fs_dir,
350
+ Some ( PathBuf :: from( "/path/to/storage" ) )
351
+ ) ;
352
+ }
353
+
354
+ #[ test]
355
+ fn test_cli_args_to_config_origin_block_padding ( ) {
356
+ let cli_args = CliArgs {
357
+ verbose : 3 ,
358
+ log_dir : Some ( "logs" . to_string ( ) ) ,
359
+ log_rotation : Some ( "DAILY" . to_string ( ) ) ,
360
+ beacon_endpoint : "http://localhost:5052" . to_string ( ) ,
361
+ beacon_client_timeout : 10 ,
362
+ poll_interval : 6 ,
363
+ listen_addr : "0.0.0.0:8000" . to_string ( ) ,
364
+ origin_block : "0x1" . to_string ( ) ,
365
+ storage_type : "fs" . to_string ( ) ,
366
+ s3_endpoint : None ,
367
+ s3_bucket : None ,
368
+ s3_path : None ,
369
+ s3_compress : None ,
370
+ fs_dir : Some ( "/path/to/storage" . to_string ( ) ) ,
371
+ } ;
372
+
373
+ let config = cli_args. to_config ( ) ;
374
+
375
+ assert_eq ! (
376
+ config. origin_block,
377
+ Hash256 :: from_slice(
378
+ & hex:: decode( "1000000000000000000000000000000000000000000000000000000000000000" )
379
+ . unwrap( )
380
+ )
381
+ ) ;
382
+ }
383
+
384
+ #[ test]
385
+ fn test_cli_args_to_config_origin_block_without_prefix ( ) {
386
+ let cli_args = CliArgs {
387
+ verbose : 3 ,
388
+ log_dir : Some ( "logs" . to_string ( ) ) ,
389
+ log_rotation : Some ( "DAILY" . to_string ( ) ) ,
390
+ beacon_endpoint : "http://localhost:5052" . to_string ( ) ,
391
+ beacon_client_timeout : 10 ,
392
+ poll_interval : 6 ,
393
+ listen_addr : "0.0.0.0:8000" . to_string ( ) ,
394
+ origin_block : "ff" . to_string ( ) ,
395
+ storage_type : "fs" . to_string ( ) ,
396
+ s3_endpoint : None ,
397
+ s3_bucket : None ,
398
+ s3_path : None ,
399
+ s3_compress : None ,
400
+ fs_dir : Some ( "/path/to/storage" . to_string ( ) ) ,
401
+ } ;
402
+
403
+ let config = cli_args. to_config ( ) ;
404
+
405
+ assert_eq ! (
406
+ config. origin_block,
407
+ Hash256 :: from_slice(
408
+ & hex:: decode( "ff00000000000000000000000000000000000000000000000000000000000000" )
409
+ . unwrap( )
410
+ )
411
+ ) ;
412
+ }
413
+ }
414
+
415
+ #[ derive( Debug , PartialEq , Eq , Clone , Default , Serialize , Deserialize ) ]
416
+ pub struct LogConfig {
417
+ log_dir : Option < PathBuf > ,
418
+ log_rotation : Option < String > ,
419
+ verbose : u8 ,
420
+ }
0 commit comments