Skip to content

Commit

Permalink
Merge pull request #271 from ralexstokes/val-tooling-cleanups
Browse files Browse the repository at this point in the history
various cleanups from #269
  • Loading branch information
ralexstokes authored Sep 28, 2023
2 parents 16947e7 + 0bb583c commit 1884b36
Show file tree
Hide file tree
Showing 2 changed files with 85 additions and 69 deletions.
143 changes: 83 additions & 60 deletions ethereum-consensus/src/bin/ec/validator/keystores.rs
Original file line number Diff line number Diff line change
Expand Up @@ -9,24 +9,35 @@ use scrypt::{
},
Params as ScryptParams, Scrypt,
};
use serde::{Deserialize, Serialize};
use serde::{Deserialize, Serialize, Serializer};
use unicode_normalization::UnicodeNormalization;
use uuid::Uuid;

pub type Passphrase = String;
type Aes128Ctr64BE = ctr::Ctr64BE<aes::Aes128>;
fn as_hex<S, D: AsRef<[u8]>>(data: D, s: S) -> Result<S::Ok, S::Error>
where
S: Serializer,
{
let encoding = hex::encode(data);
s.collect_str(&encoding)
}

fn empty_string<S, D: AsRef<[u8]>>(_data: D, s: S) -> Result<S::Ok, S::Error>
where
S: Serializer,
{
s.collect_str(&"")
}

pub type Passphrase = String;
const PASSPHRASE_LEN: usize = 32;

const VERSION: usize = 4;

const KDF_FN: &str = "scrypt";
const SCRYPT_DKLEN: usize = 32;
const SCRYPT_LOG_N: u8 = 18;
const SCRYPT_R: u32 = 8;
const SCRYPT_P: u32 = 1;

const CIPHER_FN: &str = "aes-128-ctr";
type CtrCipher = ctr::Ctr64BE<aes::Aes128>;
const AES_SIZE: usize = 16;

const CHECKSUM_FN: &str = "sha256";
Expand All @@ -37,17 +48,16 @@ struct KdfParams {
r: u32,
p: u32,
dklen: usize,
// hex encoded
salt: String,
#[serde(serialize_with = "as_hex")]
salt: Vec<u8>,
}

#[derive(Debug, Serialize, Deserialize)]
struct Kdf {
function: String,
params: KdfParams,
message: String,
#[serde(skip)]
message_bytes: Vec<u8>,
#[serde(serialize_with = "empty_string")]
message: Vec<u8>,
}

impl Kdf {
Expand All @@ -59,75 +69,68 @@ impl Kdf {
let mut passphrase = passphrase.nfkd().collect::<String>();
passphrase.retain(|c| !c.is_control());
let mut buf = vec![0u8; salt.len()];
let salt_bytes = salt.decode_b64(&mut buf).unwrap();
let salt_bytes = salt.decode_b64(&mut buf).unwrap().to_vec();

let password_hash = Scrypt
.hash_password_customized(passphrase.as_bytes(), None, None, params, &salt)
.unwrap();

let salt = hex::encode(salt_bytes);
let n = 2_usize.pow(params.log_n() as u32);
let params = KdfParams { n, r: params.r(), p: params.p(), dklen: SCRYPT_DKLEN, salt };
let message_bytes = password_hash.hash.unwrap().as_bytes().to_vec();
Self { function: KDF_FN.to_string(), params, message: "".to_string(), message_bytes }
let params =
KdfParams { n, r: params.r(), p: params.p(), dklen: SCRYPT_DKLEN, salt: salt_bytes };
let message = password_hash.hash.unwrap().as_bytes().to_vec();
Self { function: KDF_FN.to_string(), params, message }
}

fn from(passphrase: &Passphrase) -> Self {
let salt = SaltString::generate(OsRng);
let params = ScryptParams::new(SCRYPT_LOG_N, SCRYPT_R, SCRYPT_P, SCRYPT_DKLEN).unwrap();
// NOTE: `SCRYPT_DKLEN` used in `new_with_salt_and_params` matches the `recommended` params
let params = ScryptParams::recommended();

Self::new_with_salt_and_params(passphrase, salt, params)
}

fn encryption_key(&self) -> &[u8] {
&self.message_bytes[..16]
&self.message[..16]
}

fn checksum_key(&self) -> &[u8] {
&self.message_bytes[16..]
fn checksum_salt(&self) -> &[u8] {
&self.message[16..]
}
}

#[derive(Debug, Serialize, Deserialize)]
pub struct CipherParams {
// hex encoded
iv: String,
#[serde(serialize_with = "as_hex")]
iv: Vec<u8>,
}

#[derive(Debug, Serialize, Deserialize)]
pub struct Cipher {
function: String,
params: CipherParams,
// hex encoded
message: String,
#[serde(skip)]
message_bytes: Vec<u8>,
#[serde(serialize_with = "as_hex")]
message: Vec<u8>,
}

impl Cipher {
fn new_with_iv(secret: BlsSecretKey, key: &[u8], iv: &[u8]) -> Self {
let mut cipher = Aes128Ctr64BE::new_from_slices(key, iv).unwrap();
let mut cipher_text = secret.to_bytes();
cipher.apply_keystream(&mut cipher_text);

let message = hex::encode(cipher_text);
let params = CipherParams { iv: hex::encode(iv) };
Self {
function: CIPHER_FN.to_string(),
params,
message,
message_bytes: cipher_text.to_vec(),
}
fn new_with_iv(secret: BlsSecretKey, key: &[u8], iv: [u8; AES_SIZE]) -> Self {
let mut cipher = CtrCipher::new_from_slices(key, &iv).unwrap();
let mut message = secret.to_bytes().to_vec();
cipher.apply_keystream(&mut message);

let params = CipherParams { iv: iv.to_vec() };
Self { function: CIPHER_FN.to_string(), params, message }
}

fn new(secret: BlsSecretKey, key: &[u8]) -> Self {
let mut iv = [0u8; AES_SIZE];
OsRng.fill_bytes(&mut iv);
Self::new_with_iv(secret, key, &iv)
Self::new_with_iv(secret, key, iv)
}

fn cipher_text(&self) -> &[u8] {
&self.message_bytes
&self.message
}
}

Expand All @@ -138,16 +141,15 @@ struct Empty {}
pub struct Checksum {
function: String,
params: Empty,
// hex encoded
message: String,
#[serde(serialize_with = "as_hex")]
message: Vec<u8>,
}

impl Checksum {
fn new(secret: &[u8], key: &[u8]) -> Self {
let mut pre_image = key.to_vec();
fn new(secret: &[u8], salt: &[u8]) -> Self {
let mut pre_image = salt.to_vec();
pre_image.extend_from_slice(secret);
let message = hash(pre_image);
let message = hex::encode(message.as_ref());
let message = hash(pre_image).as_ref().to_vec();
Self { function: CHECKSUM_FN.to_string(), params: Empty {}, message }
}
}
Expand All @@ -163,7 +165,7 @@ impl Crypto {
fn from(passphrase: &Passphrase, key: BlsSecretKey) -> Self {
let kdf = Kdf::from(passphrase);
let cipher = Cipher::new(key, kdf.encryption_key());
let checksum = Checksum::new(cipher.cipher_text(), kdf.checksum_key());
let checksum = Checksum::new(cipher.cipher_text(), kdf.checksum_salt());
Crypto { kdf, checksum, cipher }
}
}
Expand Down Expand Up @@ -197,15 +199,30 @@ impl Keystore {
}
}

pub fn generate(keys: Vec<KeyPair>) -> (Vec<Keystore>, Vec<Passphrase>) {
keys.into_par_iter().map(Keystore::new_with_generated_passphrase).unzip()
#[derive(Debug, Serialize, Deserialize)]
pub struct KeystoreWithPassphrase {
keystore: Keystore,
passphrase: Passphrase,
}

pub fn generate(keys: Vec<KeyPair>) -> Vec<KeystoreWithPassphrase> {
keys.into_par_iter()
.map(|key_pair| {
let (keystore, passphrase) = Keystore::new_with_generated_passphrase(key_pair);
KeystoreWithPassphrase { keystore, passphrase }
})
.collect()
}

#[cfg(test)]
mod tests {
use super::*;
use base64::prelude::*;

const SCRYPT_LOG_N: u8 = 18;
const SCRYPT_R: u32 = 8;
const SCRYPT_P: u32 = 1;

// Test case from EIP-2335:
#[test]
fn test_keystore_pubkey() {
Expand All @@ -225,29 +242,35 @@ mod tests {
let passphrase = "𝔱𝔢𝔰𝔱𝔭𝔞𝔰𝔰𝔴𝔬𝔯𝔡🔑".to_string();
let secret = "000000000019d6689c085ae165831e934ff763ae46a2a6c172b3f1b60a8ce26f";
let secret_key = BlsSecretKey::try_from(hex::decode(secret).unwrap().as_ref()).unwrap();

let salt = hex::decode("d4e56740f876aef8c010b86a40d5f56745a118d0906a34e69aec8c0db1cb8fa3")
.unwrap();
let salt = &BASE64_STANDARD_NO_PAD.encode(salt);
let salt = SaltString::from_b64(salt).unwrap();
let params = ScryptParams::new(18, SCRYPT_R, SCRYPT_P, SCRYPT_DKLEN).unwrap();
let params = ScryptParams::new(SCRYPT_LOG_N, SCRYPT_R, SCRYPT_P, SCRYPT_DKLEN).unwrap();
let kdf = Kdf::new_with_salt_and_params(&passphrase, salt, params);

let iv = hex::decode("264daa3f303d7259501c93d997d84fe6").unwrap();
let cipher = Cipher::new_with_iv(secret_key, kdf.encryption_key(), &iv);
let checksum = Checksum::new(&cipher.message_bytes, kdf.checksum_key());
let cipher = Cipher::new_with_iv(secret_key, kdf.encryption_key(), iv.try_into().unwrap());

let checksum = Checksum::new(cipher.cipher_text(), kdf.checksum_salt());

assert_eq!(kdf.params.n, 262144);
assert_eq!(
&kdf.params.salt,
"d4e56740f876aef8c010b86a40d5f56745a118d0906a34e69aec8c0db1cb8fa3"
kdf.params.salt,
hex::decode("d4e56740f876aef8c010b86a40d5f56745a118d0906a34e69aec8c0db1cb8fa3")
.unwrap()
);
assert_eq!(
&checksum.message,
"d2217fe5f3e9a1e34581ef8a78f7c9928e436d36dacc5e846690a5581e8ea484"
checksum.message,
hex::decode("d2217fe5f3e9a1e34581ef8a78f7c9928e436d36dacc5e846690a5581e8ea484")
.unwrap()
);
assert_eq!(&cipher.params.iv, "264daa3f303d7259501c93d997d84fe6");
assert_eq!(cipher.params.iv, hex::decode("264daa3f303d7259501c93d997d84fe6").unwrap());
assert_eq!(
&cipher.message,
"06ae90d55fe0a6e9c5c3bc5b170827b2e5cce3929ed3f116c2811e6366dfe20f"
cipher.message,
hex::decode("06ae90d55fe0a6e9c5c3bc5b170827b2e5cce3929ed3f116c2811e6366dfe20f")
.unwrap()
);
}
}
11 changes: 2 additions & 9 deletions ethereum-consensus/src/bin/ec/validator/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -29,15 +29,8 @@ impl Command {
let mnemonic = mnemonic::recover_from_phrase(&phrase)?;
let seed = mnemonic::to_seed(mnemonic, None);
let (signing_keys, _withdrawal_keys) = keys::generate(&seed, start, end);
let (keystores, passwords) = keystores::generate(signing_keys);
let keystores_json = keystores
.iter()
.map(|keystore| serde_json::to_string_pretty(keystore).unwrap())
.collect::<Vec<_>>();
dbg!(keystores, passwords);
for keystore in keystores_json {
println!("{keystore}");
}
let keystores_with_passphrases = keystores::generate(signing_keys);
println!("{}", serde_json::to_string_pretty(&keystores_with_passphrases).unwrap());
Ok(())
}
}
Expand Down

0 comments on commit 1884b36

Please sign in to comment.