// Copyright 2019 Parity Technologies (UK) Ltd. // // Permission is hereby granted, free of charge, to any person obtaining a // copy of this software and associated documentation files (the "Software"), // to deal in the Software without restriction, including without limitation // the rights to use, copy, modify, merge, publish, distribute, sublicense, // and/or sell copies of the Software, and to permit persons to whom the // Software is furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in // all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS // OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER // DEALINGS IN THE SOFTWARE. //! Ed25519 keys. use ed25519_dalek as ed25519; use failure::Fail; use super::error::DecodingError; use zeroize::Zeroize; /// An Ed25519 keypair. pub struct Keypair(ed25519::Keypair); impl Keypair { /// Generate a new Ed25519 keypair. pub fn generate() -> Keypair { Keypair(ed25519::Keypair::generate(&mut rand::thread_rng())) } /// Encode the keypair into a byte array by concatenating the bytes /// of the secret scalar and the compressed public point, /// an informal standard for encoding Ed25519 keypairs. pub fn encode(&self) -> [u8; 64] { self.0.to_bytes() } /// Decode a keypair from the format produced by `encode`, /// zeroing the input on success. pub fn decode(kp: &mut [u8]) -> Result { ed25519::Keypair::from_bytes(kp) .map(|k| { kp.zeroize(); Keypair(k) }) .map_err(|e| DecodingError::new("Ed25519 keypair", e.compat())) } /// Sign a message using the private key of this keypair. pub fn sign(&self, msg: &[u8]) -> Vec { self.0.sign(msg).to_bytes().to_vec() } /// Get the public key of this keypair. pub fn public(&self) -> PublicKey { PublicKey(self.0.public) } /// Get the secret key of this keypair. pub fn secret(&self) -> SecretKey { SecretKey::from_bytes(&mut self.0.secret.to_bytes()) .expect("ed25519::SecretKey::from_bytes(to_bytes(k)) != k") } } impl Clone for Keypair { fn clone(&self) -> Keypair { let mut sk_bytes = self.0.secret.to_bytes(); let secret = SecretKey::from_bytes(&mut sk_bytes) .expect("ed25519::SecretKey::from_bytes(to_bytes(k)) != k").0; let public = ed25519::PublicKey::from_bytes(&self.0.public.to_bytes()) .expect("ed25519::PublicKey::from_bytes(to_bytes(k)) != k"); Keypair(ed25519::Keypair { secret, public }) } } /// Demote an Ed25519 keypair to a secret key. impl From for SecretKey { fn from(kp: Keypair) -> SecretKey { SecretKey(kp.0.secret) } } /// Promote an Ed25519 secret key into a keypair. impl From for Keypair { fn from(sk: SecretKey) -> Keypair { let secret = sk.0; let public = ed25519::PublicKey::from(&secret); Keypair(ed25519::Keypair { secret, public }) } } /// An Ed25519 public key. #[derive(PartialEq, Eq, Debug, Clone)] pub struct PublicKey(ed25519::PublicKey); impl PublicKey { /// Verify the Ed25519 signature on a message using the public key. pub fn verify(&self, msg: &[u8], sig: &[u8]) -> bool { ed25519::Signature::from_bytes(sig).map(|s| self.0.verify(msg, &s)).is_ok() } /// Encode the public key into a byte array in compressed form, i.e. /// where one coordinate is represented by a single bit. pub fn encode(&self) -> [u8; 32] { self.0.to_bytes() } /// Decode a public key from a byte array as produced by `encode`. pub fn decode(k: &[u8]) -> Result { ed25519::PublicKey::from_bytes(k) .map_err(|e| DecodingError::new("Ed25519 public key", e.compat())) .map(PublicKey) } } /// An Ed25519 secret key. pub struct SecretKey(ed25519::SecretKey); /// View the bytes of the secret key. impl AsRef<[u8]> for SecretKey { fn as_ref(&self) -> &[u8] { self.0.as_bytes() } } impl Clone for SecretKey { fn clone(&self) -> SecretKey { let mut sk_bytes = self.0.to_bytes(); Self::from_bytes(&mut sk_bytes) .expect("ed25519::SecretKey::from_bytes(to_bytes(k)) != k") } } impl SecretKey { /// Generate a new Ed25519 secret key. pub fn generate() -> SecretKey { SecretKey(ed25519::SecretKey::generate(&mut rand::thread_rng())) } /// Create an Ed25519 secret key from a byte slice, zeroing the input on success. /// If the bytes do not constitute a valid Ed25519 secret key, an error is /// returned. pub fn from_bytes(mut sk_bytes: impl AsMut<[u8]>) -> Result { let sk_bytes = sk_bytes.as_mut(); let secret = ed25519::SecretKey::from_bytes(&*sk_bytes) .map_err(|e| DecodingError::new("Ed25519 secret key", e.compat()))?; sk_bytes.zeroize(); Ok(SecretKey(secret)) } } #[cfg(test)] mod tests { use super::*; use quickcheck::*; fn eq_keypairs(kp1: &Keypair, kp2: &Keypair) -> bool { kp1.public() == kp2.public() && kp1.0.secret.as_bytes() == kp2.0.secret.as_bytes() } #[test] fn ed25519_keypair_encode_decode() { fn prop() -> bool { let kp1 = Keypair::generate(); let mut kp1_enc = kp1.encode(); let kp2 = Keypair::decode(&mut kp1_enc).unwrap(); eq_keypairs(&kp1, &kp2) && kp1_enc.iter().all(|b| *b == 0) } QuickCheck::new().tests(10).quickcheck(prop as fn() -> _); } #[test] fn ed25519_keypair_from_secret() { fn prop() -> bool { let kp1 = Keypair::generate(); let mut sk = kp1.0.secret.to_bytes(); let kp2 = Keypair::from(SecretKey::from_bytes(&mut sk).unwrap()); eq_keypairs(&kp1, &kp2) && sk == [0u8; 32] } QuickCheck::new().tests(10).quickcheck(prop as fn() -> _); } }