-
Notifications
You must be signed in to change notification settings - Fork 17
/
Copy pathcollisions.rs
59 lines (54 loc) · 1.65 KB
/
collisions.rs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
//! A simple script to check how many bytes in an address are needed to
//! distinguish accounts. The script gets the account list from the node.
use clap::AppSettings;
use concordium_rust_sdk::{
id::types::AccountAddress,
v2::{self, BlockIdentifier},
};
use futures::TryStreamExt;
use structopt::StructOpt;
#[derive(StructOpt)]
struct App {
#[structopt(
long = "node",
help = "GRPC interface of the node.",
default_value = "http://localhost:20000"
)]
endpoint: v2::Endpoint,
}
#[tokio::main(flavor = "multi_thread")]
async fn main() -> anyhow::Result<()> {
let app = {
let app = App::clap().global_setting(AppSettings::ColoredHelp);
let matches = app.get_matches();
App::from_clap(&matches)
};
let mut client = v2::Client::new(app.endpoint).await?;
let bi = client
.get_account_list(BlockIdentifier::LastFinal)
.await?
.response
.try_collect::<Vec<_>>()
.await?;
println!("There are {} accounts.", bi.len());
for i in 0..32 {
let mut tmp = bi.clone();
let compare = |l: &AccountAddress, r: &AccountAddress| {
let l_bytes: &[u8] = l.as_ref();
let r_bytes: &[u8] = r.as_ref();
l_bytes[0..i].cmp(&r_bytes[0..i])
};
tmp.sort_by(compare);
tmp.dedup_by(|l, r| compare(l, r).is_eq());
if tmp.len() < bi.len() {
println!("Clashes at prefix of length {}.", i);
} else {
println!(
"The first {} bytes of the address uniquely determine the account.",
i
);
break;
}
}
Ok(())
}