2024-06-09 13:24:44 +00:00
|
|
|
use std::{fmt::Write, fs::write};
|
|
|
|
|
|
|
|
use clap::Parser;
|
|
|
|
use tower::{Service, ServiceExt};
|
|
|
|
|
2024-06-26 21:51:06 +00:00
|
|
|
use cuprate_blockchain::{
|
2024-08-05 20:47:30 +00:00
|
|
|
config::ConfigBuilder, cuprate_database::RuntimeError, service::BlockchainReadHandle,
|
2024-06-26 21:51:06 +00:00
|
|
|
};
|
2024-07-29 00:13:08 +00:00
|
|
|
use cuprate_types::{
|
2024-08-05 20:47:30 +00:00
|
|
|
blockchain::{BlockchainReadRequest, BlockchainResponse},
|
2024-07-29 00:13:08 +00:00
|
|
|
Chain,
|
|
|
|
};
|
2024-06-09 13:24:44 +00:00
|
|
|
|
|
|
|
use cuprate_fast_sync::{hash_of_hashes, BlockId, HashOfHashes};
|
|
|
|
|
|
|
|
const BATCH_SIZE: u64 = 512;
|
|
|
|
|
|
|
|
async fn read_batch(
|
2024-08-05 20:47:30 +00:00
|
|
|
handle: &mut BlockchainReadHandle,
|
2024-06-09 13:24:44 +00:00
|
|
|
height_from: u64,
|
|
|
|
) -> Result<Vec<BlockId>, RuntimeError> {
|
|
|
|
let mut block_ids = Vec::<BlockId>::with_capacity(BATCH_SIZE as usize);
|
|
|
|
|
|
|
|
for height in height_from..(height_from + BATCH_SIZE) {
|
2024-08-05 20:47:30 +00:00
|
|
|
let request = BlockchainReadRequest::BlockHash(height, Chain::Main);
|
2024-06-09 13:24:44 +00:00
|
|
|
let response_channel = handle.ready().await?.call(request);
|
|
|
|
let response = response_channel.await?;
|
|
|
|
|
|
|
|
match response {
|
2024-08-05 20:47:30 +00:00
|
|
|
BlockchainResponse::BlockHash(block_id) => block_ids.push(block_id),
|
2024-06-09 13:24:44 +00:00
|
|
|
_ => unreachable!(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(block_ids)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn generate_hex(hashes: &[HashOfHashes]) -> String {
|
|
|
|
let mut s = String::new();
|
|
|
|
|
|
|
|
writeln!(&mut s, "[").unwrap();
|
|
|
|
|
|
|
|
for hash in hashes {
|
|
|
|
writeln!(&mut s, "\thex!(\"{}\"),", hex::encode(hash)).unwrap();
|
|
|
|
}
|
|
|
|
|
|
|
|
writeln!(&mut s, "]").unwrap();
|
|
|
|
|
|
|
|
s
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Parser)]
|
|
|
|
#[command(version, about, long_about = None)]
|
|
|
|
struct Args {
|
|
|
|
#[arg(short, long)]
|
|
|
|
height: u64,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[tokio::main]
|
|
|
|
async fn main() {
|
|
|
|
let args = Args::parse();
|
|
|
|
let height_target = args.height;
|
|
|
|
|
|
|
|
let config = ConfigBuilder::new().build();
|
|
|
|
|
2024-08-05 20:47:30 +00:00
|
|
|
let (mut read_handle, _, _) = cuprate_blockchain::service::init(config).unwrap();
|
2024-06-09 13:24:44 +00:00
|
|
|
|
|
|
|
let mut hashes_of_hashes = Vec::new();
|
|
|
|
|
|
|
|
let mut height = 0u64;
|
|
|
|
|
|
|
|
while height < height_target {
|
|
|
|
match read_batch(&mut read_handle, height).await {
|
|
|
|
Ok(block_ids) => {
|
|
|
|
let hash = hash_of_hashes(block_ids.as_slice());
|
|
|
|
hashes_of_hashes.push(hash);
|
|
|
|
}
|
|
|
|
Err(_) => {
|
|
|
|
println!("Failed to read next batch from database");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
height += BATCH_SIZE;
|
|
|
|
}
|
|
|
|
|
|
|
|
drop(read_handle);
|
|
|
|
|
|
|
|
let generated = generate_hex(&hashes_of_hashes);
|
|
|
|
write("src/data/hashes_of_hashes", generated).expect("Could not write file");
|
|
|
|
|
|
|
|
println!("Generated hashes up to block height {}", height);
|
|
|
|
}
|