From 0bb583c38f004691156da99558af9c33ba1e327b Mon Sep 17 00:00:00 2001 From: Alex Stokes Date: Thu, 28 Sep 2023 17:23:39 -0600 Subject: [PATCH] various cleanups from #269 --- .../src/bin/ec/validator/keystores.rs | 143 ++++++++++-------- .../src/bin/ec/validator/mod.rs | 11 +- 2 files changed, 85 insertions(+), 69 deletions(-) diff --git a/ethereum-consensus/src/bin/ec/validator/keystores.rs b/ethereum-consensus/src/bin/ec/validator/keystores.rs index ebed9280b..a5b0d37f4 100644 --- a/ethereum-consensus/src/bin/ec/validator/keystores.rs +++ b/ethereum-consensus/src/bin/ec/validator/keystores.rs @@ -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; +fn as_hex>(data: D, s: S) -> Result +where + S: Serializer, +{ + let encoding = hex::encode(data); + s.collect_str(&encoding) +} + +fn empty_string>(_data: D, s: S) -> Result +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; const AES_SIZE: usize = 16; const CHECKSUM_FN: &str = "sha256"; @@ -37,17 +48,16 @@ struct KdfParams { r: u32, p: u32, dklen: usize, - // hex encoded - salt: String, + #[serde(serialize_with = "as_hex")] + salt: Vec, } #[derive(Debug, Serialize, Deserialize)] struct Kdf { function: String, params: KdfParams, - message: String, - #[serde(skip)] - message_bytes: Vec, + #[serde(serialize_with = "empty_string")] + message: Vec, } impl Kdf { @@ -59,75 +69,68 @@ impl Kdf { let mut passphrase = passphrase.nfkd().collect::(); 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, } #[derive(Debug, Serialize, Deserialize)] pub struct Cipher { function: String, params: CipherParams, - // hex encoded - message: String, - #[serde(skip)] - message_bytes: Vec, + #[serde(serialize_with = "as_hex")] + message: Vec, } 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 } } @@ -138,16 +141,15 @@ struct Empty {} pub struct Checksum { function: String, params: Empty, - // hex encoded - message: String, + #[serde(serialize_with = "as_hex")] + message: Vec, } 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 } } } @@ -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 } } } @@ -197,8 +199,19 @@ impl Keystore { } } -pub fn generate(keys: Vec) -> (Vec, Vec) { - 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) -> Vec { + keys.into_par_iter() + .map(|key_pair| { + let (keystore, passphrase) = Keystore::new_with_generated_passphrase(key_pair); + KeystoreWithPassphrase { keystore, passphrase } + }) + .collect() } #[cfg(test)] @@ -206,6 +219,10 @@ 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() { @@ -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() ); } } diff --git a/ethereum-consensus/src/bin/ec/validator/mod.rs b/ethereum-consensus/src/bin/ec/validator/mod.rs index 4d8afd51b..7f79ae774 100644 --- a/ethereum-consensus/src/bin/ec/validator/mod.rs +++ b/ethereum-consensus/src/bin/ec/validator/mod.rs @@ -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::>(); - 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(()) } }