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
use core::convert::TryInto;
use sp_std::{vec, vec::Vec};
use crate::{
utils::key_encoding::{
decode_stellar_key, encode_stellar_key, ED25519_PUBLIC_KEY_BYTE_LENGTH,
ED25519_PUBLIC_KEY_VERSION_BYTE,
},
PublicKey, StellarSdkError, XdrCodec,
};
use sodalite::{sign_attached_open, Sign as Signature, SIGN_LEN};
pub trait IntoPublicKey {
fn into_public_key(self) -> Result<PublicKey, StellarSdkError>;
}
impl IntoPublicKey for PublicKey {
fn into_public_key(self) -> Result<PublicKey, StellarSdkError> {
Ok(self)
}
}
impl<T: AsRef<[u8]>> IntoPublicKey for T {
fn into_public_key(self) -> Result<PublicKey, StellarSdkError> {
PublicKey::from_encoding(self)
}
}
impl PublicKey {
pub fn from_binary(binary: [u8; ED25519_PUBLIC_KEY_BYTE_LENGTH]) -> Self {
PublicKey::PublicKeyTypeEd25519(binary)
}
pub fn as_binary(&self) -> &[u8; ED25519_PUBLIC_KEY_BYTE_LENGTH] {
match self {
PublicKey::PublicKeyTypeEd25519(key) => key,
}
}
pub fn into_binary(self) -> [u8; ED25519_PUBLIC_KEY_BYTE_LENGTH] {
*self.as_binary()
}
pub fn from_encoding<T: AsRef<[u8]>>(encoded_key: T) -> Result<Self, StellarSdkError> {
let decoded_key = decode_stellar_key(encoded_key, ED25519_PUBLIC_KEY_VERSION_BYTE)?;
Ok(Self::from_binary(decoded_key))
}
pub fn to_encoding(&self) -> Vec<u8> {
let key = self.as_binary();
encode_stellar_key(key, ED25519_PUBLIC_KEY_VERSION_BYTE)
}
pub fn get_signature_hint(&self) -> [u8; 4] {
let account_id_xdr = Self::PublicKeyTypeEd25519(self.as_binary().clone()).to_xdr();
account_id_xdr[account_id_xdr.len() - 4..]
.try_into()
.unwrap()
}
pub fn verify_signature<T: AsRef<[u8]>>(&self, message: T, signature: &Signature) -> bool {
let message = message.as_ref();
let mut signed_message: Vec<u8> = Vec::with_capacity(message.len() + SIGN_LEN);
signed_message.extend_from_slice(signature);
signed_message.extend_from_slice(message);
sign_attached_open(
&mut vec![0; message.len() + SIGN_LEN],
&signed_message,
self.as_binary(),
)
.is_ok()
}
}