feat(wallet): implement HD private keys & encrypted wrapper (#358)

This commit is contained in:
Harper 2023-12-23 17:24:32 +01:00 committed by GitHub
parent 8b13646680
commit 550eac147b
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
6 changed files with 413 additions and 175 deletions

View file

@ -18,6 +18,7 @@ pallas-crypto = { path = "../pallas-crypto", version = "=0.20.0" }
pallas-primitives = { path = "../pallas-primitives", version = "=0.20.0" }
pallas-traverse = { path = "../pallas-traverse", version = "=0.20.0" }
pallas-addresses = { path = "../pallas-addresses", version = "=0.20.0" }
pallas-wallet = { path = "../pallas-wallet", version = "=0.20.0" }
serde = { version = "1.0.188", features = ["derive"] }
serde_json = "1.0.107"
thiserror = "1.0.44"

View file

@ -4,6 +4,7 @@ use pallas_crypto::{
key::ed25519,
};
use pallas_primitives::{babbage, Fragment};
use pallas_wallet::PrivateKey;
use std::{collections::HashMap, ops::Deref};
@ -608,14 +609,14 @@ pub struct BuiltTransaction {
}
impl BuiltTransaction {
pub fn sign(mut self, secret_key: ed25519::SecretKey) -> Result<Self, TxBuilderError> {
let pubkey: [u8; 32] = secret_key
pub fn sign(mut self, private_key: PrivateKey) -> Result<Self, TxBuilderError> {
let pubkey: [u8; 32] = private_key
.public_key()
.as_ref()
.try_into()
.map_err(|_| TxBuilderError::MalformedKey)?;
let signature: [u8; 64] = secret_key.sign(self.tx_hash.0).as_ref().try_into().unwrap();
let signature: [u8; ed25519::Signature::SIZE] = private_key.sign(self.tx_hash.0).as_ref().try_into().unwrap();
match self.era {
BuilderEra::Babbage => {

View file

@ -16,3 +16,4 @@ ed25519-bip32 = "0.4.1"
bip39 = { version = "2.0.0", features = ["rand_core"] }
cryptoxide = "0.4.4"
bech32 = "0.9.1"
rand = "0.8.5"

192
pallas-wallet/src/hd.rs Normal file
View file

@ -0,0 +1,192 @@
use bech32::{FromBase32, ToBase32};
use bip39::rand_core::{CryptoRng, RngCore};
use bip39::{Language, Mnemonic};
use cryptoxide::{hmac::Hmac, pbkdf2::pbkdf2, sha2::Sha512};
use ed25519_bip32::{self, XPrv, XPub, XPRV_SIZE};
use pallas_crypto::key::ed25519;
use crate::{Error, PrivateKey};
/// Ed25519-BIP32 HD Private Key
#[derive(Debug, PartialEq, Eq)]
pub struct Bip32PrivateKey(ed25519_bip32::XPrv);
impl Bip32PrivateKey {
const BECH32_HRP: &'static str = "xprv";
pub fn generate<T: RngCore + CryptoRng>(mut rng: T) -> Self {
let mut buf = [0u8; XPRV_SIZE];
rng.fill_bytes(&mut buf);
let xprv = XPrv::normalize_bytes_force3rd(buf);
Self(xprv)
}
pub fn generate_with_mnemonic<T: RngCore + CryptoRng>(
mut rng: T,
password: String,
) -> (Self, Mnemonic) {
let mut buf = [0u8; 64];
rng.fill_bytes(&mut buf);
let bip39 = Mnemonic::generate_in_with(&mut rng, Language::English, 24).unwrap();
let entropy = bip39.clone().to_entropy();
let mut pbkdf2_result = [0; XPRV_SIZE];
const ITER: u32 = 4096; // TODO: BIP39 says 2048, CML uses 4096?
let mut mac = Hmac::new(Sha512::new(), password.as_bytes());
pbkdf2(&mut mac, &entropy, ITER, &mut pbkdf2_result);
(Self(XPrv::normalize_bytes_force3rd(pbkdf2_result)), bip39)
}
pub fn from_bytes(bytes: [u8; 96]) -> Result<Self, Error> {
XPrv::from_bytes_verified(bytes)
.map(Self)
.map_err(Error::Xprv)
}
pub fn as_bytes(&self) -> Vec<u8> {
self.0.as_ref().to_vec()
}
pub fn from_bip39_mnenomic(mnemonic: String, password: String) -> Result<Self, Error> {
let bip39 = Mnemonic::parse(mnemonic).map_err(Error::Mnemonic)?;
let entropy = bip39.to_entropy();
let mut pbkdf2_result = [0; XPRV_SIZE];
const ITER: u32 = 4096; // TODO: BIP39 says 2048, CML uses 4096?
let mut mac = Hmac::new(Sha512::new(), password.as_bytes());
pbkdf2(&mut mac, &entropy, ITER, &mut pbkdf2_result);
Ok(Self(XPrv::normalize_bytes_force3rd(pbkdf2_result)))
}
pub fn derive(&self, index: u32) -> Self {
Self(self.0.derive(ed25519_bip32::DerivationScheme::V2, index))
}
pub fn to_ed25519_private_key(&self) -> PrivateKey {
PrivateKey::Extended(self.0.extended_secret_key().into())
}
pub fn to_public(&self) -> Bip32PublicKey {
Bip32PublicKey(self.0.public())
}
pub fn chain_code(&self) -> [u8; 32] {
*self.0.chain_code()
}
pub fn to_bech32(&self) -> String {
bech32::encode(
Self::BECH32_HRP,
self.as_bytes().to_base32(),
bech32::Variant::Bech32,
)
.unwrap()
}
pub fn from_bech32(bech32: String) -> Result<Self, Error> {
let (hrp, data, _) = bech32::decode(&bech32).map_err(Error::InvalidBech32)?;
if hrp != Self::BECH32_HRP {
Err(Error::InvalidBech32Hrp)
} else {
let data = Vec::<u8>::from_base32(&data).map_err(Error::InvalidBech32)?;
Self::from_bytes(data.try_into().map_err(|_| Error::UnexpectedBech32Length)?)
}
}
}
/// Ed25519-BIP32 HD Public Key
#[derive(Debug, PartialEq, Eq)]
pub struct Bip32PublicKey(ed25519_bip32::XPub);
impl Bip32PublicKey {
const BECH32_HRP: &'static str = "xpub";
pub fn from_bytes(bytes: [u8; 64]) -> Self {
Self(XPub::from_bytes(bytes))
}
pub fn as_bytes(&self) -> Vec<u8> {
self.0.as_ref().to_vec()
}
pub fn derive(&self, index: u32) -> Result<Self, Error> {
self.0
.derive(ed25519_bip32::DerivationScheme::V2, index)
.map(Self)
.map_err(Error::DerivationError)
}
pub fn to_ed25519_pubkey(&self) -> ed25519::PublicKey {
self.0.public_key().into()
}
pub fn chain_code(&self) -> [u8; 32] {
*self.0.chain_code()
}
pub fn to_bech32(&self) -> String {
bech32::encode(
Self::BECH32_HRP,
self.as_bytes().to_base32(),
bech32::Variant::Bech32,
)
.unwrap()
}
pub fn from_bech32(bech32: String) -> Result<Self, Error> {
let (hrp, data, _) = bech32::decode(&bech32).map_err(Error::InvalidBech32)?;
if hrp != Self::BECH32_HRP {
Err(Error::InvalidBech32Hrp)
} else {
let data = Vec::<u8>::from_base32(&data).map_err(Error::InvalidBech32)?;
Ok(Self::from_bytes(
data.try_into().map_err(|_| Error::UnexpectedBech32Length)?,
))
}
}
}
#[cfg(test)]
mod test {
use bip39::rand_core::OsRng;
use super::{Bip32PrivateKey, Bip32PublicKey};
#[test]
fn mnemonic_roundtrip() {
let (xprv, mne) = Bip32PrivateKey::generate_with_mnemonic(OsRng, "".into());
let xprv_from_mne =
Bip32PrivateKey::from_bip39_mnenomic(mne.to_string(), "".into()).unwrap();
assert_eq!(xprv, xprv_from_mne)
}
#[test]
fn bech32_roundtrip() {
let xprv = Bip32PrivateKey::generate(OsRng);
let xprv_bech32 = xprv.to_bech32();
let decoded_xprv = Bip32PrivateKey::from_bech32(xprv_bech32).unwrap();
assert_eq!(xprv, decoded_xprv);
let xpub = xprv.to_public();
let xpub_bech32 = xpub.to_bech32();
let decoded_xpub = Bip32PublicKey::from_bech32(xpub_bech32).unwrap();
assert_eq!(xpub, decoded_xpub)
}
}

View file

@ -1,13 +1,18 @@
use bech32::{FromBase32, ToBase32};
use bip39::rand_core::{CryptoRng, RngCore};
use bip39::{Language, Mnemonic};
use cryptoxide::{hmac::Hmac, pbkdf2::pbkdf2, sha2::Sha512};
use ed25519_bip32::{self, XPrv, XPub, XPRV_SIZE};
use pallas_crypto::key::ed25519::{self};
use ed25519_bip32;
use pallas_crypto::key::ed25519::{PublicKey, SecretKey, SecretKeyExtended, Signature};
use thiserror::Error;
pub mod hd;
pub mod wrapper;
#[derive(Error, Debug)]
pub enum Error {
/// Private key wrapper data of unexpected length
#[error("Wrapped private key data invalid length")]
WrapperDataInvalidSize,
/// Failed to decrypt private key wrapper data
#[error("Failed to decrypt private key wrapper data")]
WrapperDataFailedToDecrypt,
/// Unexpected bech32 HRP prefix
#[error("Unexpected bech32 HRP prefix")]
InvalidBech32Hrp,
@ -28,186 +33,59 @@ pub enum Error {
DerivationError(ed25519_bip32::DerivationError),
}
/// ED25519-BIP32 HD Private Key
#[derive(Debug, PartialEq, Eq)]
pub struct Bip32PrivateKey(ed25519_bip32::XPrv);
/// A standard or extended Ed25519 secret key
pub enum PrivateKey {
Normal(SecretKey),
Extended(SecretKeyExtended),
}
impl Bip32PrivateKey {
const BECH32_HRP: &'static str = "xprv";
pub fn generate<T: RngCore + CryptoRng>(mut rng: T) -> Self {
let mut buf = [0u8; XPRV_SIZE];
rng.fill_bytes(&mut buf);
let xprv = XPrv::normalize_bytes_force3rd(buf);
Self(xprv)
impl PrivateKey {
pub fn len(&self) -> usize {
match self {
Self::Normal(_) => SecretKey::SIZE,
Self::Extended(_) => SecretKeyExtended::SIZE,
}
}
pub fn generate_with_mnemonic<T: RngCore + CryptoRng>(
mut rng: T,
password: String,
) -> (Self, Mnemonic) {
let mut buf = [0u8; 64];
rng.fill_bytes(&mut buf);
let bip39 = Mnemonic::generate_in_with(&mut rng, Language::English, 24).unwrap();
let entropy = bip39.clone().to_entropy();
let mut pbkdf2_result = [0; XPRV_SIZE];
const ITER: u32 = 4096; // TODO: BIP39 says 2048, CML uses 4096?
let mut mac = Hmac::new(Sha512::new(), password.as_bytes());
pbkdf2(&mut mac, &entropy, ITER, &mut pbkdf2_result);
(Self(XPrv::normalize_bytes_force3rd(pbkdf2_result)), bip39)
pub fn public_key(&self) -> PublicKey {
match self {
Self::Normal(x) => x.public_key(),
Self::Extended(x) => x.public_key(),
}
}
pub fn from_bytes(bytes: [u8; 96]) -> Result<Self, Error> {
XPrv::from_bytes_verified(bytes)
.map(Self)
.map_err(Error::Xprv)
pub fn sign<T>(&self, msg: T) -> Signature
where
T: AsRef<[u8]>,
{
match self {
Self::Normal(x) => x.sign(msg),
Self::Extended(x) => x.sign(msg),
}
}
pub fn as_bytes(&self) -> Vec<u8> {
self.0.as_ref().to_vec()
}
pub fn from_bip39_mnenomic(mnemonic: String, password: String) -> Result<Self, Error> {
let bip39 = Mnemonic::parse(mnemonic).map_err(Error::Mnemonic)?;
let entropy = bip39.to_entropy();
let mut pbkdf2_result = [0; XPRV_SIZE];
const ITER: u32 = 4096; // TODO: BIP39 says 2048, CML uses 4096?
let mut mac = Hmac::new(Sha512::new(), password.as_bytes());
pbkdf2(&mut mac, &entropy, ITER, &mut pbkdf2_result);
Ok(Self(XPrv::normalize_bytes_force3rd(pbkdf2_result)))
}
pub fn derive(&self, index: u32) -> Self {
Self(self.0.derive(ed25519_bip32::DerivationScheme::V2, index))
}
pub fn to_ed25519_privkey(&self) -> ed25519::SecretKeyExtended {
self.0.extended_secret_key().into()
}
pub fn to_public(&self) -> Bip32PublicKey {
Bip32PublicKey(self.0.public())
}
pub fn chain_code(&self) -> [u8; 32] {
*self.0.chain_code()
}
pub fn to_bech32(&self) -> String {
bech32::encode(
Self::BECH32_HRP,
self.as_bytes().to_base32(),
bech32::Variant::Bech32,
)
.unwrap()
}
pub fn from_bech32(bech32: String) -> Result<Self, Error> {
let (hrp, data, _) = bech32::decode(&bech32).map_err(Error::InvalidBech32)?;
if hrp != Self::BECH32_HRP {
Err(Error::InvalidBech32Hrp)
} else {
let data = Vec::<u8>::from_base32(&data).map_err(Error::InvalidBech32)?;
Self::from_bytes(data.try_into().map_err(|_| Error::UnexpectedBech32Length)?)
match self {
Self::Normal(x) => {
let bytes: [u8; SecretKey::SIZE] = x.clone().into();
bytes.to_vec()
}
Self::Extended(x) => {
let bytes: [u8; SecretKeyExtended::SIZE] = x.clone().into();
bytes.to_vec()
}
}
}
}
/// ED25519-BIP32 HD Public Key
#[derive(Debug, PartialEq, Eq)]
pub struct Bip32PublicKey(ed25519_bip32::XPub);
impl Bip32PublicKey {
const BECH32_HRP: &'static str = "xpub";
pub fn from_bytes(bytes: [u8; 64]) -> Self {
Self(XPub::from_bytes(bytes))
}
pub fn as_bytes(&self) -> Vec<u8> {
self.0.as_ref().to_vec()
}
pub fn derive(&self, index: u32) -> Result<Self, Error> {
self.0
.derive(ed25519_bip32::DerivationScheme::V2, index)
.map(Self)
.map_err(Error::DerivationError)
}
pub fn to_ed25519_pubkey(&self) -> ed25519::PublicKey {
self.0.public_key().into()
}
pub fn chain_code(&self) -> [u8; 32] {
*self.0.chain_code()
}
pub fn to_bech32(&self) -> String {
bech32::encode(
Self::BECH32_HRP,
self.as_bytes().to_base32(),
bech32::Variant::Bech32,
)
.unwrap()
}
pub fn from_bech32(bech32: String) -> Result<Self, Error> {
let (hrp, data, _) = bech32::decode(&bech32).map_err(Error::InvalidBech32)?;
if hrp != Self::BECH32_HRP {
Err(Error::InvalidBech32Hrp)
} else {
let data = Vec::<u8>::from_base32(&data).map_err(Error::InvalidBech32)?;
Ok(Self::from_bytes(
data.try_into().map_err(|_| Error::UnexpectedBech32Length)?,
))
}
impl From<SecretKey> for PrivateKey {
fn from(key: SecretKey) -> Self {
PrivateKey::Normal(key)
}
}
#[cfg(test)]
mod test {
use bip39::rand_core::OsRng;
use crate::{Bip32PrivateKey, Bip32PublicKey};
#[test]
fn mnemonic_roundtrip() {
let (xprv, mne) = Bip32PrivateKey::generate_with_mnemonic(OsRng, "".into());
let xprv_from_mne =
Bip32PrivateKey::from_bip39_mnenomic(mne.to_string(), "".into()).unwrap();
assert_eq!(xprv, xprv_from_mne)
}
#[test]
fn bech32_roundtrip() {
let xprv = Bip32PrivateKey::generate(OsRng);
let xprv_bech32 = xprv.to_bech32();
let decoded_xprv = Bip32PrivateKey::from_bech32(xprv_bech32).unwrap();
assert_eq!(xprv, decoded_xprv);
let xpub = xprv.to_public();
let xpub_bech32 = xpub.to_bech32();
let decoded_xpub = Bip32PublicKey::from_bech32(xpub_bech32).unwrap();
assert_eq!(xpub, decoded_xpub)
impl From<SecretKeyExtended> for PrivateKey {
fn from(key: SecretKeyExtended) -> Self {
PrivateKey::Extended(key)
}
}

View file

@ -0,0 +1,165 @@
use cryptoxide::chacha20poly1305::ChaCha20Poly1305;
use cryptoxide::kdf::argon2;
use pallas_crypto::key::ed25519::{SecretKey, SecretKeyExtended};
use rand::{CryptoRng, RngCore};
use crate::{Error, PrivateKey};
const ITERATIONS: u32 = 2500;
const VERSION_SIZE: usize = 1;
const SALT_SIZE: usize = 16;
const NONCE_SIZE: usize = 12;
const TAG_SIZE: usize = 16;
pub fn encrypt_private_key<Rng>(mut rng: Rng, private_key: PrivateKey, password: &String) -> Vec<u8>
where
Rng: RngCore + CryptoRng,
{
let salt = {
let mut salt = [0u8; SALT_SIZE];
rng.fill_bytes(&mut salt);
salt
};
let sym_key: [u8; 32] = argon2::argon2(
&argon2::Params::argon2d().iterations(ITERATIONS).unwrap(),
password.as_bytes(),
&salt,
&[],
&[],
);
let nonce = {
let mut nonce = [0u8; NONCE_SIZE];
rng.fill_bytes(&mut nonce);
nonce
};
let mut chacha20 = ChaCha20Poly1305::new(&sym_key, &nonce, &[]);
let data_size = private_key.len();
let (ciphertext, ct_tag) = {
let mut ciphertext = vec![0u8; data_size];
let mut ct_tag = [0u8; 16];
chacha20.encrypt(&private_key.as_bytes(), &mut ciphertext, &mut ct_tag);
(ciphertext, ct_tag)
};
// (version || salt || nonce || tag || ciphertext)
let mut out = Vec::with_capacity(VERSION_SIZE + SALT_SIZE + NONCE_SIZE + TAG_SIZE + data_size);
out.push(1);
out.extend_from_slice(&salt);
out.extend_from_slice(&nonce);
out.extend_from_slice(&ct_tag);
out.extend_from_slice(&ciphertext);
out
}
#[allow(unused)]
pub fn decrypt_private_key(password: &String, data: Vec<u8>) -> Result<PrivateKey, Error> {
let data_len_without_ct = VERSION_SIZE + SALT_SIZE + NONCE_SIZE + TAG_SIZE;
let ciphertext_len = if data.len() == (data_len_without_ct + SecretKey::SIZE) {
SecretKey::SIZE
} else if data.len() == (data_len_without_ct + SecretKeyExtended::SIZE) {
SecretKeyExtended::SIZE
} else {
return Err(Error::WrapperDataInvalidSize);
};
let mut cursor = 0;
let _version = &data[cursor];
cursor += VERSION_SIZE;
let salt = &data[cursor..cursor + SALT_SIZE];
cursor += SALT_SIZE;
let nonce = &data[cursor..cursor + NONCE_SIZE];
cursor += NONCE_SIZE;
let tag = &data[cursor..cursor + TAG_SIZE];
cursor += TAG_SIZE;
let ciphertext = &data[cursor..cursor + ciphertext_len];
let sym_key: [u8; 32] = argon2::argon2(
&argon2::Params::argon2d().iterations(ITERATIONS).unwrap(),
password.as_bytes(),
salt,
&[],
&[],
);
let mut chacha20 = ChaCha20Poly1305::new(&sym_key, nonce, &[]);
match ciphertext_len {
SecretKey::SIZE => {
let mut plaintext = [0u8; SecretKey::SIZE];
if chacha20.decrypt(ciphertext, &mut plaintext, tag) {
let secret_key: SecretKey = plaintext.into();
Ok(secret_key.into())
} else {
Err(Error::WrapperDataFailedToDecrypt)
}
}
SecretKeyExtended::SIZE => {
let mut plaintext = [0u8; SecretKeyExtended::SIZE];
if chacha20.decrypt(ciphertext, &mut plaintext, tag) {
let secret_key: SecretKeyExtended = plaintext.into();
Ok(secret_key.into())
} else {
Err(Error::WrapperDataFailedToDecrypt)
}
}
_ => unreachable!(),
}
}
#[cfg(test)]
mod tests {
use pallas_crypto::key::ed25519::{SecretKey, SecretKeyExtended};
use rand::rngs::OsRng;
use crate::{
wrapper::{decrypt_private_key, encrypt_private_key},
PrivateKey,
};
#[test]
fn private_key_encryption_roundtrip() {
let password = "hunter123";
// --- standard
let private_key = PrivateKey::Normal(SecretKey::new(OsRng));
let private_key_bytes = private_key.as_bytes();
let encrypted_priv_key = encrypt_private_key(OsRng, private_key, &password.into());
let decrypted_privkey = decrypt_private_key(&password.into(), encrypted_priv_key).unwrap();
assert_eq!(private_key_bytes, decrypted_privkey.as_bytes());
// --- extended
let private_key = PrivateKey::Extended(SecretKeyExtended::new(OsRng));
let private_key_bytes = private_key.as_bytes();
let encrypted_priv_key = encrypt_private_key(OsRng, private_key, &password.into());
let decrypted_privkey = decrypt_private_key(&password.into(), encrypted_priv_key).unwrap();
assert_eq!(private_key_bytes, decrypted_privkey.as_bytes())
}
}