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
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
use crate::core_types::StdSignature;
use crate::keys::PublicKey;
use bitcoin::secp256k1::Secp256k1;
use bitcoin::secp256k1::{All, Message};
use bitcoin::util::bip32::{ExtendedPrivKey, IntoDerivationPath};
use bitcoin::Network;
use crypto::sha2::Sha256;
use crypto::digest::Digest;
use hkd32::mnemonic::{Phrase, Seed};
use crate::errors::TerraRustAPIError;
use rand_core::OsRng;
pub static LUNA_COIN_TYPE: u32 = 330;
pub struct PrivateKey {
#[allow(missing_docs)]
pub account: u32,
#[allow(missing_docs)]
pub index: u32,
#[allow(missing_docs)]
pub coin_type: u32,
mnemonic: Option<Phrase>,
#[allow(dead_code)]
root_private_key: ExtendedPrivKey,
private_key: ExtendedPrivKey,
}
impl PrivateKey {
pub fn new(secp: &Secp256k1<All>) -> anyhow::Result<PrivateKey> {
let phrase =
hkd32::mnemonic::Phrase::random(&mut OsRng, hkd32::mnemonic::Language::English);
PrivateKey::gen_private_key_phrase(secp, phrase, 0, 0, LUNA_COIN_TYPE, "")
}
pub fn new_seed(secp: &Secp256k1<All>, seed_phrase: &str) -> anyhow::Result<PrivateKey> {
let phrase =
hkd32::mnemonic::Phrase::random(&mut OsRng, hkd32::mnemonic::Language::English);
PrivateKey::gen_private_key_phrase(secp, phrase, 0, 0, LUNA_COIN_TYPE, seed_phrase)
}
pub fn from_words(secp: &Secp256k1<All>, words: &str) -> anyhow::Result<PrivateKey> {
match hkd32::mnemonic::Phrase::new(words, hkd32::mnemonic::Language::English) {
Ok(phrase) => {
PrivateKey::gen_private_key_phrase(secp, phrase, 0, 0, LUNA_COIN_TYPE, "")
}
Err(_) => Err(TerraRustAPIError::Phrasing.into()),
}
}
pub fn from_words_seed(
secp: &Secp256k1<All>,
words: &str,
seed_pass: &str,
) -> anyhow::Result<PrivateKey> {
match hkd32::mnemonic::Phrase::new(words, hkd32::mnemonic::Language::English) {
Ok(phrase) => {
PrivateKey::gen_private_key_phrase(secp, phrase, 0, 0, LUNA_COIN_TYPE, seed_pass)
}
Err(_) => Err(TerraRustAPIError::Phrasing.into()),
}
}
pub fn public_key(&self, secp: &Secp256k1<All>) -> PublicKey {
let x = &self.private_key.private_key.public_key(secp);
PublicKey::from_bitcoin_public_key(x)
}
fn gen_private_key_phrase(
secp: &Secp256k1<All>,
phrase: Phrase,
account: u32,
index: u32,
coin_type: u32,
seed_phrase: &str,
) -> anyhow::Result<PrivateKey> {
let seed = phrase.to_seed(seed_phrase);
let root_private_key =
ExtendedPrivKey::new_master(Network::Bitcoin, seed.as_bytes()).unwrap();
let path = format!("m/44'/{}'/{}'/0/{}", coin_type, account, index);
let derivation_path = path.into_derivation_path()?;
let private_key = root_private_key.derive_priv(secp, &derivation_path)?;
Ok(PrivateKey {
account,
index,
coin_type,
mnemonic: Some(phrase),
root_private_key,
private_key,
})
}
pub fn words(&self) -> Option<&str> {
self.mnemonic.as_ref().map(|phrase| phrase.phrase())
}
pub fn sign(&self, secp: &Secp256k1<All>, blob: &str) -> anyhow::Result<StdSignature> {
let pub_k = &self.private_key.private_key.public_key(secp);
let priv_k = self.private_key.private_key.key;
let mut sha = Sha256::new();
let mut sha_result: [u8; 32] = [0; 32];
sha.input_str(blob);
sha.result(&mut sha_result);
let message: Message = Message::from_slice(&sha_result)?;
let signature = secp.sign(&message, &priv_k);
let sig: StdSignature = StdSignature::create(&signature.serialize_compact(), pub_k);
Ok(sig)
}
#[allow(dead_code)]
pub(crate) fn seed(&self, passwd: &str) -> Option<Seed> {
self.mnemonic.as_ref().map(|phrase| phrase.to_seed(passwd))
}
}
#[cfg(test)]
mod tst {
use super::*;
#[test]
pub fn tst_gen_mnemonic() -> anyhow::Result<()> {
let s = Secp256k1::new();
PrivateKey::new(&s).and_then(|_| Ok(()))
}
#[test]
pub fn tst_words() -> anyhow::Result<()> {
let str_1 = "notice oak worry limit wrap speak medal online prefer cluster roof addict wrist behave treat actual wasp year salad speed social layer crew genius";
let seed_1 = "a2ae8846397b55d266af35acdbb18ba1d005f7ddbdd4ca7a804df83352eaf373f274ba0dc8ac1b2b25f19dfcb7fa8b30a240d2c6039d88963defc2f626003b2f";
let s = Secp256k1::new();
let pk = PrivateKey::from_words(&s, str_1)?;
assert_eq!(hex::encode(pk.seed("").unwrap().as_bytes()), seed_1);
match pk.words() {
Some(words) => {
assert_eq!(words, str_1);
Ok(())
}
None => Err(TerraRustAPIError::MissingPhrase.into()),
}
}
#[test]
pub fn tst_root_priv_key() -> anyhow::Result<()> {
let str_1 = "wonder caution square unveil april art add hover spend smile proud admit modify old copper throw crew happy nature luggage reopen exhibit ordinary napkin";
let secp = Secp256k1::new();
let pk = PrivateKey::from_words(&secp, str_1)?;
let root_key = "xprv9s21ZrQH143K2ep3BpYRRMjSqjLHZAPAzxfVVS3NBuGKBVtCrK3C8mE8TcmTjYnLm7SJxdLigDFWGAMnctKxc3p5QKNWXdprcFSQzGzQqTW";
assert_eq!(pk.root_private_key.to_string(), root_key);
let derived_key = "4804e2bdce36d413206ccf47cc4c64db2eff924e7cc9e90339fa7579d2bd9d5b";
assert_eq!(pk.private_key.private_key.key.to_string(), derived_key);
Ok(())
}
#[test]
pub fn tst_words_to_pub() -> anyhow::Result<()> {
let str_1 = "wonder caution square unveil april art add hover spend smile proud admit modify old copper throw crew happy nature luggage reopen exhibit ordinary napkin";
let secp = Secp256k1::new();
let pk = PrivateKey::from_words(&secp, str_1)?;
let pub_k = pk.public_key(&secp);
let account = pub_k.account()?;
assert_eq!(&account, "terra1jnzv225hwl3uxc5wtnlgr8mwy6nlt0vztv3qqm");
assert_eq!(
&pub_k.operator_address_public_key()?,
"terravaloperpub1addwnpepqt8ha594svjn3nvfk4ggfn5n8xd3sm3cz6ztxyugwcuqzsuuhhfq5y7accr"
);
assert_eq!(
&pub_k.application_public_key()?,
"terrapub1addwnpepqt8ha594svjn3nvfk4ggfn5n8xd3sm3cz6ztxyugwcuqzsuuhhfq5nwzrf9"
);
Ok(())
}
#[test]
pub fn test_sign() -> anyhow::Result<()> {
let str_1 = "island relax shop such yellow opinion find know caught erode blue dolphin behind coach tattoo light focus snake common size analyst imitate employ walnut";
let secp = Secp256k1::new();
let pk = PrivateKey::from_words(&secp, str_1)?;
let _pub_k = pk.public_key(&secp);
let to_sign = r#"{"account_number":"45","chain_id":"columbus-3-testnet","fee":{"amount":[{"amount":"698","denom":"uluna"}],"gas":"46467"},"memo":"","msgs":[{"type":"bank/MsgSend","value":{"amount":[{"amount":"100000000","denom":"uluna"}],"from_address":"terra1n3g37dsdlv7ryqftlkef8mhgqj4ny7p8v78lg7","to_address":"terra1wg2mlrxdmnnkkykgqg4znky86nyrtc45q336yv"}}],"sequence":"0"}"#;
let sig = pk.sign(&secp, to_sign)?;
assert_eq!(
sig.pub_key.value,
"AiMzHaA2bvnDXfHzkjMM+vkSE/p0ymBtAFKUnUtQAeXe"
);
assert_eq!(sig.signature, "FJKAXRxNB5ruqukhVqZf3S/muZEUmZD10fVmWycdVIxVWiCXXFsUy2VY2jINEOUGNwfrqEZsT2dUfAvWj8obLg==");
Ok(())
}
}