1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
extern crate nimiq_keys as keys;
extern crate nimiq_hash as hash;
use keys::{PrivateKey, PublicKey};
use hash::hmac::*;
use byteorder::{BigEndian, WriteBytesExt};
use beserial::Serialize;
use hash::Sha512Hash;
use keys::Address;
use regex::Regex;
use std::borrow::Cow;
#[derive(Clone, Debug, Eq, PartialEq)]
pub struct ExtendedPrivateKey {
key: PrivateKey,
chain_code: [u8; ExtendedPrivateKey::CHAIN_CODE_SIZE],
}
const B_CURVE: [u8; 12] = [101, 100, 50, 53, 53, 49, 57, 32, 115, 101, 101, 100];
impl ExtendedPrivateKey {
pub const CHAIN_CODE_SIZE: usize = 32;
pub fn from_seed(seed: Vec<u8>) -> Self {
let hash = compute_hmac_sha512(&B_CURVE, seed.as_slice());
ExtendedPrivateKey::from(hash)
}
pub fn is_valid_path(path: &str) -> bool {
let re = Regex::new(r"^m(/[0-9]+')*$").unwrap();
if !re.is_match(&path) {
return false;
}
path.split("/").skip(1).all(|segment| segment.trim_end_matches("'").parse::<u32>().is_ok())
}
pub fn derive(&self, mut index: u32) -> Option<Self> {
if index < 0x80000000 {
index = index.checked_add(0x80000000)?;
}
let mut data = Vec::<u8>::with_capacity(1 + PrivateKey::SIZE + 4);
data.write_u8(0).ok()?;
self.key.serialize(&mut data).ok()?;
data.write_u32::<BigEndian>(index).ok()?;
let hash = compute_hmac_sha512(&self.chain_code, data.as_slice());
Some(ExtendedPrivateKey::from(hash))
}
pub fn derive_path(&self, path: &str) -> Option<Self> {
if !ExtendedPrivateKey::is_valid_path(path) {
return None;
}
let mut derived_key: Cow<ExtendedPrivateKey> = Cow::Borrowed(self);
for segment in path.split("/").skip(1) {
derived_key = Cow::Owned(derived_key.derive(segment.trim_end_matches("'").parse::<u32>().unwrap())?);
}
Some(derived_key.into_owned())
}
pub fn get_chain_code(&self) -> &[u8] {
&self.chain_code
}
pub fn into_private_key(self) -> PrivateKey {
self.key
}
pub fn to_public_key(&self) -> PublicKey {
PublicKey::from(&self.key)
}
pub fn to_address(&self) -> Address {
let pub_key = PublicKey::from(&self.key);
Address::from(&pub_key)
}
}
impl From<Sha512Hash> for ExtendedPrivateKey {
fn from(hash: Sha512Hash) -> Self {
let mut private_key: [u8; PrivateKey::SIZE] = Default::default();
private_key.copy_from_slice(&hash.as_bytes()[..32]);
let mut chain_code: [u8; ExtendedPrivateKey::CHAIN_CODE_SIZE] = Default::default();
chain_code.copy_from_slice(&hash.as_bytes()[32..]);
ExtendedPrivateKey {
key: PrivateKey::from(private_key),
chain_code
}
}
}
impl From<ExtendedPrivateKey> for PrivateKey {
fn from(key: ExtendedPrivateKey) -> Self {
key.into_private_key()
}
}
impl<'a> From<&'a ExtendedPrivateKey> for Address {
fn from(key: &'a ExtendedPrivateKey) -> Self {
key.to_address()
}
}
impl<'a> From<&'a ExtendedPrivateKey> for PublicKey {
fn from(key: &'a ExtendedPrivateKey) -> Self {
key.to_public_key()
}
}