2017-05-25 02:08:39 +03:00
|
|
|
// Copyright 2016 The Grin Developers
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
|
|
|
/// Key derivation scheme used by Grin to build chains of private keys
|
|
|
|
/// in its wallet logic. Largely inspired by bitcoin's BIP32.
|
|
|
|
|
|
|
|
use std::{error, fmt};
|
2017-09-22 19:45:06 +03:00
|
|
|
use std::cmp::min;
|
2017-05-25 02:08:39 +03:00
|
|
|
|
|
|
|
use byteorder::{ByteOrder, BigEndian};
|
2017-09-27 01:51:45 +03:00
|
|
|
use blake2::blake2b::blake2b;
|
2017-05-25 02:08:39 +03:00
|
|
|
use secp::Secp256k1;
|
|
|
|
use secp::key::SecretKey;
|
|
|
|
|
|
|
|
/// An ExtKey error
|
|
|
|
#[derive(Copy, PartialEq, Eq, Clone, Debug)]
|
|
|
|
pub enum Error {
|
|
|
|
/// The size of the seed is invalid
|
|
|
|
InvalidSeedSize,
|
|
|
|
InvalidSliceSize,
|
|
|
|
InvalidExtendedKey,
|
|
|
|
}
|
|
|
|
|
|
|
|
// Passthrough Debug to Display, since errors should be user-visible
|
|
|
|
impl fmt::Display for Error {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> Result<(), fmt::Error> {
|
|
|
|
f.write_str(error::Error::description(self))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl error::Error for Error {
|
|
|
|
fn cause(&self) -> Option<&error::Error> {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
|
|
|
|
fn description(&self) -> &str {
|
|
|
|
match *self {
|
|
|
|
Error::InvalidSeedSize => "wallet: seed isn't of size 128, 256 or 512",
|
|
|
|
// TODO change when ser. ext. size is fixed
|
|
|
|
Error::InvalidSliceSize => "wallet: serialized extended key must be of size 73",
|
|
|
|
Error::InvalidExtendedKey => "wallet: the given serialized extended key is invalid",
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-22 19:45:06 +03:00
|
|
|
#[derive(Serialize, Deserialize, Clone)]
|
|
|
|
pub struct Fingerprint([u8; 4]);
|
|
|
|
|
|
|
|
impl Fingerprint {
|
|
|
|
fn from_bytes(bytes: &[u8]) -> Fingerprint {
|
|
|
|
let mut fingerprint = [0; 4];
|
|
|
|
for i in 0..min(4, bytes.len()) {
|
|
|
|
fingerprint[i] = bytes[i];
|
|
|
|
}
|
|
|
|
Fingerprint(fingerprint)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn zero() -> Fingerprint {
|
|
|
|
Fingerprint([0; 4])
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl PartialEq for Fingerprint {
|
|
|
|
fn eq(&self, other: &Self) -> bool {
|
|
|
|
self.0.as_ref() == other.0.as_ref()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ::std::fmt::Display for Fingerprint {
|
|
|
|
fn fmt(&self, f: &mut ::std::fmt::Formatter) -> ::std::fmt::Result {
|
|
|
|
for i in self.0.iter().cloned() {
|
|
|
|
try!(write!(f, "{:02x}", i));
|
|
|
|
}
|
|
|
|
write!(f, "")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ::std::fmt::Debug for Fingerprint {
|
|
|
|
fn fmt(&self, f: &mut ::std::fmt::Formatter) -> ::std::fmt::Result {
|
|
|
|
try!(write!(f, "{}(", stringify!(Fingerprint)));
|
|
|
|
for i in self.0.iter().cloned() {
|
|
|
|
try!(write!(f, "{:02x}", i));
|
|
|
|
}
|
|
|
|
write!(f, ")")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-27 01:51:45 +03:00
|
|
|
#[derive(Serialize, Deserialize, Clone)]
|
2017-09-22 19:45:06 +03:00
|
|
|
pub struct Identifier([u8; 20]);
|
|
|
|
|
|
|
|
impl Identifier {
|
|
|
|
fn from_bytes(bytes: &[u8]) -> Identifier {
|
|
|
|
let mut identifier = [0; 20];
|
|
|
|
for i in 0..min(20, bytes.len()) {
|
|
|
|
identifier[i] = bytes[i];
|
|
|
|
}
|
|
|
|
Identifier(identifier)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn fingerprint(&self) -> Fingerprint {
|
|
|
|
Fingerprint::from_bytes(&self.0)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-27 01:51:45 +03:00
|
|
|
impl PartialEq for Identifier {
|
|
|
|
fn eq(&self, other: &Self) -> bool {
|
|
|
|
self.0.as_ref() == other.0.as_ref()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-22 19:45:06 +03:00
|
|
|
impl ::std::fmt::Debug for Identifier {
|
|
|
|
fn fmt(&self, f: &mut ::std::fmt::Formatter) -> ::std::fmt::Result {
|
|
|
|
try!(write!(f, "{}(", stringify!(Identifier)));
|
|
|
|
for i in self.0.iter().cloned() {
|
|
|
|
try!(write!(f, "{:02x}", i));
|
|
|
|
}
|
|
|
|
write!(f, ")")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-05-25 02:08:39 +03:00
|
|
|
/// An ExtendedKey is a secret key which can be used to derive new
|
|
|
|
/// secret keys to blind the commitment of a transaction output.
|
|
|
|
/// To be usable, a secret key should have an amount assigned to it,
|
|
|
|
/// but when the key is derived, the amount is not known and must be
|
|
|
|
/// given.
|
|
|
|
#[derive(Debug, Clone)]
|
|
|
|
pub struct ExtendedKey {
|
|
|
|
/// Depth of the extended key
|
|
|
|
pub depth: u8,
|
|
|
|
/// Child number of the key
|
|
|
|
pub n_child: u32,
|
|
|
|
/// Parent key's fingerprint
|
2017-09-22 19:45:06 +03:00
|
|
|
pub fingerprint: Fingerprint,
|
2017-05-25 02:08:39 +03:00
|
|
|
/// Code of the derivation chain
|
|
|
|
pub chaincode: [u8; 32],
|
|
|
|
/// Actual private key
|
|
|
|
pub key: SecretKey,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ExtendedKey {
|
|
|
|
/// Creates a new extended key from a serialized one
|
|
|
|
pub fn from_slice(secp: &Secp256k1, slice: &[u8]) -> Result<ExtendedKey, Error> {
|
|
|
|
// TODO change when ser. ext. size is fixed
|
|
|
|
if slice.len() != 73 {
|
|
|
|
return Err(Error::InvalidSliceSize);
|
|
|
|
}
|
|
|
|
let depth: u8 = slice[0];
|
2017-09-22 19:45:06 +03:00
|
|
|
let fingerprint = Fingerprint::from_bytes(&slice[1..5]);
|
2017-05-25 02:08:39 +03:00
|
|
|
let n_child = BigEndian::read_u32(&slice[5..9]);
|
|
|
|
let mut chaincode: [u8; 32] = [0; 32];
|
|
|
|
(&mut chaincode).copy_from_slice(&slice[9..41]);
|
|
|
|
let secret_key = match SecretKey::from_slice(secp, &slice[41..73]) {
|
|
|
|
Ok(key) => key,
|
|
|
|
Err(_) => return Err(Error::InvalidExtendedKey),
|
|
|
|
};
|
|
|
|
|
|
|
|
Ok(ExtendedKey {
|
|
|
|
depth: depth,
|
|
|
|
fingerprint: fingerprint,
|
|
|
|
n_child: n_child,
|
|
|
|
chaincode: chaincode,
|
|
|
|
key: secret_key,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Creates a new extended master key from a seed
|
|
|
|
pub fn from_seed(secp: &Secp256k1, seed: &[u8]) -> Result<ExtendedKey, Error> {
|
|
|
|
match seed.len() {
|
2017-09-27 01:51:45 +03:00
|
|
|
16 | 32 | 64 => (),
|
2017-05-25 02:08:39 +03:00
|
|
|
_ => return Err(Error::InvalidSeedSize),
|
|
|
|
}
|
|
|
|
|
2017-09-27 01:51:45 +03:00
|
|
|
// let mut derived: [u8; 64] = [0; 64];
|
|
|
|
// hmac.raw_result(&mut derived);
|
|
|
|
|
|
|
|
let derived = blake2b(64, b"Mimble seed", seed);
|
2017-05-25 02:08:39 +03:00
|
|
|
|
|
|
|
let mut chaincode: [u8; 32] = [0; 32];
|
2017-09-27 01:51:45 +03:00
|
|
|
(&mut chaincode).copy_from_slice(&derived.as_bytes()[32..]);
|
2017-05-25 02:08:39 +03:00
|
|
|
// TODO Error handling
|
2017-09-27 01:51:45 +03:00
|
|
|
let secret_key = SecretKey::from_slice(&secp, &derived.as_bytes()[0..32])
|
2017-05-25 02:08:39 +03:00
|
|
|
.expect("Error generating from seed");
|
|
|
|
|
|
|
|
let mut ext_key = ExtendedKey {
|
|
|
|
depth: 0,
|
2017-09-22 19:45:06 +03:00
|
|
|
fingerprint: Fingerprint::zero(),
|
2017-05-25 02:08:39 +03:00
|
|
|
n_child: 0,
|
|
|
|
chaincode: chaincode,
|
|
|
|
key: secret_key,
|
|
|
|
};
|
|
|
|
|
2017-09-22 19:45:06 +03:00
|
|
|
ext_key.fingerprint = ext_key.identifier().fingerprint();
|
2017-05-25 02:08:39 +03:00
|
|
|
|
2017-05-29 06:21:29 +03:00
|
|
|
Ok(ext_key)
|
2017-05-25 02:08:39 +03:00
|
|
|
}
|
|
|
|
|
2017-09-27 01:51:45 +03:00
|
|
|
/// Return the identifier of the key
|
|
|
|
/// which is the blake2b hash (20 bit digest)
|
2017-09-22 19:45:06 +03:00
|
|
|
pub fn identifier(&self) -> Identifier {
|
2017-09-27 01:51:45 +03:00
|
|
|
let identifier = blake2b(20, &[], &self.key[..]);
|
|
|
|
Identifier::from_bytes(&identifier.as_bytes())
|
2017-05-25 02:08:39 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Derive an extended key from an extended key
|
|
|
|
pub fn derive(&self, secp: &Secp256k1, n: u32) -> Result<ExtendedKey, Error> {
|
|
|
|
let mut n_bytes: [u8; 4] = [0; 4];
|
|
|
|
BigEndian::write_u32(&mut n_bytes, n);
|
2017-09-27 01:51:45 +03:00
|
|
|
let mut seed = self.key[..].to_vec();
|
|
|
|
seed.extend_from_slice(&n_bytes);
|
2017-05-25 02:08:39 +03:00
|
|
|
|
2017-09-27 01:51:45 +03:00
|
|
|
let derived = blake2b(64, &self.chaincode[..], &seed[..]);
|
2017-05-25 02:08:39 +03:00
|
|
|
|
2017-09-27 01:51:45 +03:00
|
|
|
let mut secret_key = SecretKey::from_slice(&secp, &derived.as_bytes()[0..32])
|
2017-05-25 02:08:39 +03:00
|
|
|
.expect("Error deriving key");
|
|
|
|
secret_key.add_assign(secp, &self.key)
|
|
|
|
.expect("Error deriving key");
|
|
|
|
// TODO check if key != 0 ?
|
|
|
|
|
|
|
|
let mut chain_code: [u8; 32] = [0; 32];
|
2017-09-27 01:51:45 +03:00
|
|
|
(&mut chain_code).clone_from_slice(&derived.as_bytes()[32..]);
|
2017-05-25 02:08:39 +03:00
|
|
|
|
|
|
|
Ok(ExtendedKey {
|
|
|
|
depth: self.depth + 1,
|
2017-09-22 19:45:06 +03:00
|
|
|
fingerprint: self.identifier().fingerprint(),
|
2017-05-25 02:08:39 +03:00
|
|
|
n_child: n,
|
|
|
|
chaincode: chain_code,
|
|
|
|
key: secret_key,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod test {
|
|
|
|
use secp::Secp256k1;
|
|
|
|
use secp::key::SecretKey;
|
2017-09-27 01:51:45 +03:00
|
|
|
use super::{ExtendedKey, Fingerprint, Identifier};
|
2017-09-24 07:41:35 +03:00
|
|
|
use util;
|
|
|
|
|
|
|
|
fn from_hex(hex_str: &str) -> Vec<u8> {
|
|
|
|
util::from_hex(hex_str.to_string()).unwrap()
|
|
|
|
}
|
2017-05-25 02:08:39 +03:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn extkey_from_seed() {
|
|
|
|
// TODO More test vectors
|
|
|
|
let s = Secp256k1::new();
|
2017-09-24 07:41:35 +03:00
|
|
|
let seed = from_hex("000102030405060708090a0b0c0d0e0f");
|
2017-05-25 02:08:39 +03:00
|
|
|
let extk = ExtendedKey::from_seed(&s, &seed.as_slice()).unwrap();
|
|
|
|
let sec =
|
2017-09-27 01:51:45 +03:00
|
|
|
from_hex("c3f5ae520f474b390a637de4669c84d0ed9bbc21742577fac930834d3c3083dd");
|
2017-05-25 02:08:39 +03:00
|
|
|
let secret_key = SecretKey::from_slice(&s, sec.as_slice()).unwrap();
|
|
|
|
let chaincode =
|
2017-09-27 01:51:45 +03:00
|
|
|
from_hex("e7298e68452b0c6d54837670896e1aee76b118075150d90d4ee416ece106ae72");
|
|
|
|
let identifier = from_hex("942b6c0bd43bdcb24f3edfe7fadbc77054ecc4f2");
|
|
|
|
let fingerprint = from_hex("942b6c0b");
|
2017-05-25 02:08:39 +03:00
|
|
|
let depth = 0;
|
|
|
|
let n_child = 0;
|
|
|
|
assert_eq!(extk.key, secret_key);
|
2017-09-27 01:51:45 +03:00
|
|
|
assert_eq!(extk.identifier(), Identifier::from_bytes(identifier.as_slice()));
|
2017-09-22 19:45:06 +03:00
|
|
|
assert_eq!(extk.fingerprint, Fingerprint::from_bytes(fingerprint.as_slice()));
|
2017-09-27 01:51:45 +03:00
|
|
|
assert_eq!(
|
|
|
|
extk.identifier().fingerprint(),
|
|
|
|
Fingerprint::from_bytes(fingerprint.as_slice())
|
|
|
|
);
|
2017-05-25 02:08:39 +03:00
|
|
|
assert_eq!(extk.chaincode, chaincode.as_slice());
|
|
|
|
assert_eq!(extk.depth, depth);
|
|
|
|
assert_eq!(extk.n_child, n_child);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn extkey_derivation() {
|
2017-08-23 02:05:56 +03:00
|
|
|
// TODO More test vectors
|
2017-05-25 02:08:39 +03:00
|
|
|
let s = Secp256k1::new();
|
2017-09-24 07:41:35 +03:00
|
|
|
let seed = from_hex("000102030405060708090a0b0c0d0e0f");
|
2017-05-25 02:08:39 +03:00
|
|
|
let extk = ExtendedKey::from_seed(&s, &seed.as_slice()).unwrap();
|
|
|
|
let derived = extk.derive(&s, 0).unwrap();
|
|
|
|
let sec =
|
2017-09-27 01:51:45 +03:00
|
|
|
from_hex("d75f70beb2bd3b56f9b064087934bdedee98e4b5aae6280c58b4eff38847888f");
|
2017-05-25 02:08:39 +03:00
|
|
|
let secret_key = SecretKey::from_slice(&s, sec.as_slice()).unwrap();
|
|
|
|
let chaincode =
|
2017-09-27 01:51:45 +03:00
|
|
|
from_hex("243cb881e1549e714db31d23af45540b13ad07941f64a786bbf3313b4de1df52");
|
|
|
|
let fingerprint = from_hex("942b6c0b");
|
|
|
|
let identifier = from_hex("8b011f14345f3f0071e85f6eec116de1e575ea10");
|
|
|
|
let identifier_fingerprint = from_hex("8b011f14");
|
2017-05-25 02:08:39 +03:00
|
|
|
let depth = 1;
|
|
|
|
let n_child = 0;
|
|
|
|
assert_eq!(derived.key, secret_key);
|
2017-09-27 01:51:45 +03:00
|
|
|
assert_eq!(derived.identifier(), Identifier::from_bytes(identifier.as_slice()));
|
2017-09-22 19:45:06 +03:00
|
|
|
assert_eq!(derived.fingerprint, Fingerprint::from_bytes(fingerprint.as_slice()));
|
2017-09-27 01:51:45 +03:00
|
|
|
assert_eq!(
|
|
|
|
derived.identifier().fingerprint(),
|
|
|
|
Fingerprint::from_bytes(identifier_fingerprint.as_slice())
|
|
|
|
);
|
2017-05-25 02:08:39 +03:00
|
|
|
assert_eq!(derived.chaincode, chaincode.as_slice());
|
|
|
|
assert_eq!(derived.depth, depth);
|
|
|
|
assert_eq!(derived.n_child, n_child);
|
|
|
|
}
|
|
|
|
}
|