2018-03-05 22:33:44 +03:00
|
|
|
// Copyright 2018 The Grin Developers
|
2017-10-03 03:02:31 +03:00
|
|
|
//
|
|
|
|
// 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.
|
|
|
|
|
2018-06-08 08:21:54 +03:00
|
|
|
/// Implementation of the Keychain trait based on an extended key derivation
|
|
|
|
/// scheme.
|
2018-09-18 11:39:45 +03:00
|
|
|
use rand::distributions::Alphanumeric;
|
2017-10-03 03:02:31 +03:00
|
|
|
use rand::{thread_rng, Rng};
|
2017-10-16 20:11:01 +03:00
|
|
|
use std::collections::HashMap;
|
2017-11-10 18:12:15 +03:00
|
|
|
use std::sync::{Arc, RwLock};
|
2017-10-03 03:02:31 +03:00
|
|
|
|
|
|
|
use blake2;
|
|
|
|
|
2018-06-08 08:21:54 +03:00
|
|
|
use extkey;
|
|
|
|
use types::{BlindSum, BlindingFactor, Error, Identifier, Keychain};
|
|
|
|
use util::logger::LOGGER;
|
|
|
|
use util::secp::key::SecretKey;
|
|
|
|
use util::secp::pedersen::Commitment;
|
|
|
|
use util::secp::{self, Message, Secp256k1, Signature};
|
2018-02-28 20:56:09 +03:00
|
|
|
|
2017-10-03 03:02:31 +03:00
|
|
|
#[derive(Clone, Debug)]
|
2018-06-08 08:21:54 +03:00
|
|
|
pub struct ExtKeychain {
|
2017-10-03 03:02:31 +03:00
|
|
|
secp: Secp256k1,
|
|
|
|
extkey: extkey::ExtendedKey,
|
2017-10-16 20:11:01 +03:00
|
|
|
key_overrides: HashMap<Identifier, SecretKey>,
|
2017-11-10 18:12:15 +03:00
|
|
|
key_derivation_cache: Arc<RwLock<HashMap<Identifier, u32>>>,
|
2017-10-03 03:02:31 +03:00
|
|
|
}
|
|
|
|
|
2018-06-08 08:21:54 +03:00
|
|
|
impl Keychain for ExtKeychain {
|
|
|
|
fn from_seed(seed: &[u8]) -> Result<ExtKeychain, Error> {
|
2017-10-03 03:02:31 +03:00
|
|
|
let secp = secp::Secp256k1::with_caps(secp::ContextFlag::Commit);
|
|
|
|
let extkey = extkey::ExtendedKey::from_seed(&secp, seed)?;
|
2018-06-08 08:21:54 +03:00
|
|
|
let keychain = ExtKeychain {
|
2017-10-03 03:02:31 +03:00
|
|
|
secp: secp,
|
|
|
|
extkey: extkey,
|
2017-10-16 20:11:01 +03:00
|
|
|
key_overrides: HashMap::new(),
|
2017-11-10 18:12:15 +03:00
|
|
|
key_derivation_cache: Arc::new(RwLock::new(HashMap::new())),
|
2017-10-03 03:02:31 +03:00
|
|
|
};
|
|
|
|
Ok(keychain)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// For testing - probably not a good idea to use outside of tests.
|
2018-06-08 08:21:54 +03:00
|
|
|
fn from_random_seed() -> Result<ExtKeychain, Error> {
|
2018-09-18 11:39:45 +03:00
|
|
|
let seed: String = thread_rng().sample_iter(&Alphanumeric).take(16).collect();
|
2017-10-03 03:02:31 +03:00
|
|
|
let seed = blake2::blake2b::blake2b(32, &[], seed.as_bytes());
|
2018-06-08 08:21:54 +03:00
|
|
|
ExtKeychain::from_seed(seed.as_bytes())
|
|
|
|
}
|
|
|
|
|
|
|
|
fn root_key_id(&self) -> Identifier {
|
|
|
|
self.extkey.root_key_id.clone()
|
2017-10-03 03:02:31 +03:00
|
|
|
}
|
|
|
|
|
2018-06-08 08:21:54 +03:00
|
|
|
fn derive_key_id(&self, derivation: u32) -> Result<Identifier, Error> {
|
2018-01-25 23:19:32 +03:00
|
|
|
let child_key = self.extkey.derive(&self.secp, derivation)?;
|
|
|
|
Ok(child_key.key_id)
|
2017-10-03 03:02:31 +03:00
|
|
|
}
|
|
|
|
|
2018-06-08 08:21:54 +03:00
|
|
|
fn derived_key(&self, key_id: &Identifier) -> Result<SecretKey, Error> {
|
2017-11-10 18:12:15 +03:00
|
|
|
// first check our overrides and just return the key if we have one in there
|
2017-10-16 20:11:01 +03:00
|
|
|
if let Some(key) = self.key_overrides.get(key_id) {
|
2018-03-04 03:19:54 +03:00
|
|
|
trace!(
|
|
|
|
LOGGER,
|
|
|
|
"... Derived Key (using override) key_id: {}",
|
|
|
|
key_id
|
|
|
|
);
|
2017-10-16 20:11:01 +03:00
|
|
|
return Ok(*key);
|
2017-10-12 06:35:40 +03:00
|
|
|
}
|
2017-10-07 20:38:41 +03:00
|
|
|
|
2018-01-25 23:19:32 +03:00
|
|
|
let child_key = self.derived_child_key(key_id)?;
|
|
|
|
Ok(child_key.key)
|
2018-01-23 15:14:06 +03:00
|
|
|
}
|
|
|
|
|
2018-06-08 08:21:54 +03:00
|
|
|
fn commit(&self, amount: u64, key_id: &Identifier) -> Result<Commitment, Error> {
|
|
|
|
let skey = self.derived_key(key_id)?;
|
|
|
|
let commit = self.secp.commit(amount, skey)?;
|
|
|
|
Ok(commit)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn commit_with_key_index(&self, amount: u64, derivation: u32) -> Result<Commitment, Error> {
|
|
|
|
let child_key = self.derived_key_from_index(derivation)?;
|
|
|
|
let commit = self.secp.commit(amount, child_key.key)?;
|
|
|
|
Ok(commit)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn blind_sum(&self, blind_sum: &BlindSum) -> Result<BlindingFactor, Error> {
|
|
|
|
let mut pos_keys: Vec<SecretKey> = blind_sum
|
|
|
|
.positive_key_ids
|
|
|
|
.iter()
|
|
|
|
.filter_map(|k| self.derived_key(&k).ok())
|
|
|
|
.collect();
|
|
|
|
|
|
|
|
let mut neg_keys: Vec<SecretKey> = blind_sum
|
|
|
|
.negative_key_ids
|
|
|
|
.iter()
|
|
|
|
.filter_map(|k| self.derived_key(&k).ok())
|
|
|
|
.collect();
|
|
|
|
|
2018-09-04 12:58:26 +03:00
|
|
|
pos_keys.extend(
|
|
|
|
&blind_sum
|
|
|
|
.positive_blinding_factors
|
|
|
|
.iter()
|
|
|
|
.filter_map(|b| b.secret_key(&self.secp).ok())
|
|
|
|
.collect::<Vec<SecretKey>>(),
|
|
|
|
);
|
2018-06-08 08:21:54 +03:00
|
|
|
|
2018-09-04 12:58:26 +03:00
|
|
|
neg_keys.extend(
|
|
|
|
&blind_sum
|
|
|
|
.negative_blinding_factors
|
|
|
|
.iter()
|
|
|
|
.filter_map(|b| b.secret_key(&self.secp).ok())
|
|
|
|
.collect::<Vec<SecretKey>>(),
|
|
|
|
);
|
2018-06-08 08:21:54 +03:00
|
|
|
|
|
|
|
let sum = self.secp.blind_sum(pos_keys, neg_keys)?;
|
|
|
|
Ok(BlindingFactor::from_secret_key(sum))
|
|
|
|
}
|
|
|
|
|
|
|
|
fn sign(&self, msg: &Message, key_id: &Identifier) -> Result<Signature, Error> {
|
|
|
|
let skey = self.derived_key(key_id)?;
|
|
|
|
let sig = self.secp.sign(msg, &skey)?;
|
|
|
|
Ok(sig)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn sign_with_blinding(
|
|
|
|
&self,
|
|
|
|
msg: &Message,
|
|
|
|
blinding: &BlindingFactor,
|
|
|
|
) -> Result<Signature, Error> {
|
|
|
|
let skey = &blinding.secret_key(&self.secp)?;
|
|
|
|
let sig = self.secp.sign(msg, &skey)?;
|
|
|
|
Ok(sig)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn secp(&self) -> &Secp256k1 {
|
|
|
|
&self.secp
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ExtKeychain {
|
|
|
|
// For tests and burn only, associate a key identifier with a known secret key.
|
|
|
|
pub fn burn_enabled(keychain: &ExtKeychain, burn_key_id: &Identifier) -> ExtKeychain {
|
|
|
|
let mut key_overrides = HashMap::new();
|
|
|
|
key_overrides.insert(
|
|
|
|
burn_key_id.clone(),
|
|
|
|
SecretKey::from_slice(&keychain.secp, &[1; 32]).unwrap(),
|
|
|
|
);
|
|
|
|
ExtKeychain {
|
|
|
|
key_overrides: key_overrides,
|
|
|
|
..keychain.clone()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-25 23:19:32 +03:00
|
|
|
fn derived_child_key(&self, key_id: &Identifier) -> Result<extkey::ChildKey, Error> {
|
2018-01-23 15:14:06 +03:00
|
|
|
trace!(LOGGER, "Derived Key by key_id: {}", key_id);
|
|
|
|
|
2017-11-10 18:12:15 +03:00
|
|
|
// then check the derivation cache to see if we have previously derived this key
|
|
|
|
// if so use the derivation from the cache to derive the key
|
|
|
|
{
|
|
|
|
let cache = self.key_derivation_cache.read().unwrap();
|
|
|
|
if let Some(derivation) = cache.get(key_id) {
|
2018-03-04 03:19:54 +03:00
|
|
|
trace!(
|
|
|
|
LOGGER,
|
|
|
|
"... Derived Key (cache hit) key_id: {}, derivation: {}",
|
|
|
|
key_id,
|
|
|
|
derivation
|
|
|
|
);
|
|
|
|
return Ok(self.derived_key_from_index(*derivation)?);
|
2017-11-10 18:12:15 +03:00
|
|
|
}
|
|
|
|
}
|
2017-11-07 19:48:37 +03:00
|
|
|
|
2017-11-10 18:12:15 +03:00
|
|
|
// otherwise iterate over a large number of derivations looking for our key
|
|
|
|
// cache the resulting derivations by key_id for faster lookup later
|
2017-12-19 04:33:44 +03:00
|
|
|
// TODO - remove hard limit (within reason)
|
|
|
|
// TODO - do we benefit here if we track our max known n_child?
|
2017-11-10 18:12:15 +03:00
|
|
|
{
|
|
|
|
let mut cache = self.key_derivation_cache.write().unwrap();
|
2017-12-19 04:33:44 +03:00
|
|
|
for i in 1..100_000 {
|
2018-01-25 23:19:32 +03:00
|
|
|
let child_key = self.extkey.derive(&self.secp, i)?;
|
|
|
|
// let child_key_id = extkey.identifier(&self.secp)?;
|
|
|
|
|
|
|
|
if !cache.contains_key(&child_key.key_id) {
|
|
|
|
trace!(
|
|
|
|
LOGGER,
|
|
|
|
"... Derived Key (cache miss) key_id: {}, derivation: {}",
|
|
|
|
child_key.key_id,
|
|
|
|
child_key.n_child,
|
|
|
|
);
|
|
|
|
cache.insert(child_key.key_id.clone(), child_key.n_child);
|
2017-11-10 18:12:15 +03:00
|
|
|
}
|
|
|
|
|
2018-01-25 23:19:32 +03:00
|
|
|
if child_key.key_id == *key_id {
|
|
|
|
return Ok(child_key);
|
2017-11-10 18:12:15 +03:00
|
|
|
}
|
2017-10-07 20:38:41 +03:00
|
|
|
}
|
|
|
|
}
|
2017-11-10 18:12:15 +03:00
|
|
|
|
2018-03-04 03:19:54 +03:00
|
|
|
Err(Error::KeyDerivation(format!(
|
|
|
|
"failed to derive child_key for {:?}",
|
|
|
|
key_id
|
|
|
|
)))
|
2017-10-03 03:02:31 +03:00
|
|
|
}
|
|
|
|
|
2017-11-10 18:12:15 +03:00
|
|
|
// if we know the derivation index we can just straight to deriving the key
|
2018-03-04 03:19:54 +03:00
|
|
|
fn derived_key_from_index(&self, derivation: u32) -> Result<extkey::ChildKey, Error> {
|
2017-11-10 18:12:15 +03:00
|
|
|
trace!(LOGGER, "Derived Key (fast) by derivation: {}", derivation);
|
2018-01-25 23:19:32 +03:00
|
|
|
let child_key = self.extkey.derive(&self.secp, derivation)?;
|
2018-03-04 03:19:54 +03:00
|
|
|
return Ok(child_key);
|
2017-11-07 19:48:37 +03:00
|
|
|
}
|
2017-10-03 03:02:31 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod test {
|
2018-06-08 08:21:54 +03:00
|
|
|
use keychain::ExtKeychain;
|
|
|
|
use types::{BlindSum, BlindingFactor, Keychain};
|
2017-11-01 02:20:55 +03:00
|
|
|
use util::secp;
|
2018-02-02 17:51:55 +03:00
|
|
|
use util::secp::key::SecretKey;
|
|
|
|
|
2017-10-03 03:02:31 +03:00
|
|
|
#[test]
|
|
|
|
fn test_key_derivation() {
|
2018-06-08 08:21:54 +03:00
|
|
|
let keychain = ExtKeychain::from_random_seed().unwrap();
|
2017-11-09 22:26:45 +03:00
|
|
|
let secp = keychain.secp();
|
2017-10-03 03:02:31 +03:00
|
|
|
|
2017-10-13 07:45:07 +03:00
|
|
|
// use the keychain to derive a "key_id" based on the underlying seed
|
|
|
|
let key_id = keychain.derive_key_id(1).unwrap();
|
2017-10-03 03:02:31 +03:00
|
|
|
|
|
|
|
let msg_bytes = [0; 32];
|
|
|
|
let msg = secp::Message::from_slice(&msg_bytes[..]).unwrap();
|
|
|
|
|
|
|
|
// now create a zero commitment using the key on the keychain associated with
|
2018-01-17 06:03:40 +03:00
|
|
|
// the key_id
|
2017-10-13 07:45:07 +03:00
|
|
|
let commit = keychain.commit(0, &key_id).unwrap();
|
2017-10-03 03:02:31 +03:00
|
|
|
|
|
|
|
// now check we can use our key to verify a signature from this zero commitment
|
2017-10-13 07:45:07 +03:00
|
|
|
let sig = keychain.sign(&msg, &key_id).unwrap();
|
2017-10-03 03:02:31 +03:00
|
|
|
secp.verify_from_commit(&msg, &sig, &commit).unwrap();
|
|
|
|
}
|
2017-10-06 00:40:46 +03:00
|
|
|
|
2018-02-02 17:51:55 +03:00
|
|
|
// We plan to "offset" the key used in the kernel commitment
|
|
|
|
// so we are going to be doing some key addition/subtraction.
|
|
|
|
// This test is mainly to demonstrate that idea that summing commitments
|
|
|
|
// and summing the keys used to commit to 0 have the same result.
|
|
|
|
#[test]
|
|
|
|
fn secret_key_addition() {
|
2018-06-08 08:21:54 +03:00
|
|
|
let keychain = ExtKeychain::from_random_seed().unwrap();
|
2018-02-02 17:51:55 +03:00
|
|
|
|
|
|
|
let skey1 = SecretKey::from_slice(
|
|
|
|
&keychain.secp,
|
|
|
|
&[
|
2018-03-04 03:19:54 +03:00
|
|
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
|
|
|
0, 0, 0, 1,
|
2018-02-02 17:51:55 +03:00
|
|
|
],
|
|
|
|
).unwrap();
|
|
|
|
|
|
|
|
let skey2 = SecretKey::from_slice(
|
|
|
|
&keychain.secp,
|
|
|
|
&[
|
2018-03-04 03:19:54 +03:00
|
|
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
|
|
|
0, 0, 0, 2,
|
2018-02-02 17:51:55 +03:00
|
|
|
],
|
|
|
|
).unwrap();
|
|
|
|
|
|
|
|
// adding secret keys 1 and 2 to give secret key 3
|
|
|
|
let mut skey3 = skey1.clone();
|
|
|
|
let _ = skey3.add_assign(&keychain.secp, &skey2).unwrap();
|
|
|
|
|
|
|
|
// create commitments for secret keys 1, 2 and 3
|
|
|
|
// all committing to the value 0 (which is what we do for tx_kernels)
|
|
|
|
let commit_1 = keychain.secp.commit(0, skey1).unwrap();
|
|
|
|
let commit_2 = keychain.secp.commit(0, skey2).unwrap();
|
|
|
|
let commit_3 = keychain.secp.commit(0, skey3).unwrap();
|
|
|
|
|
|
|
|
// now sum commitments for keys 1 and 2
|
2018-03-04 03:19:54 +03:00
|
|
|
let sum = keychain
|
|
|
|
.secp
|
|
|
|
.commit_sum(vec![commit_1.clone(), commit_2.clone()], vec![])
|
|
|
|
.unwrap();
|
2018-02-02 17:51:55 +03:00
|
|
|
|
|
|
|
// confirm the commitment for key 3 matches the sum of the commitments 1 and 2
|
|
|
|
assert_eq!(sum, commit_3);
|
|
|
|
|
|
|
|
// now check we can sum keys up using keychain.blind_sum()
|
|
|
|
// in the same way (convenience function)
|
|
|
|
assert_eq!(
|
2018-03-04 03:19:54 +03:00
|
|
|
keychain
|
2018-09-04 12:58:26 +03:00
|
|
|
.blind_sum(
|
|
|
|
&BlindSum::new()
|
|
|
|
.add_blinding_factor(BlindingFactor::from_secret_key(skey1))
|
|
|
|
.add_blinding_factor(BlindingFactor::from_secret_key(skey2))
|
|
|
|
)
|
2018-03-04 03:19:54 +03:00
|
|
|
.unwrap(),
|
2018-02-13 18:35:30 +03:00
|
|
|
BlindingFactor::from_secret_key(skey3),
|
2018-02-02 17:51:55 +03:00
|
|
|
);
|
|
|
|
}
|
2017-10-03 03:02:31 +03:00
|
|
|
}
|