Impl From<[u8; 32]>/Display/FromStr for Sapling SpendingKey, including network field
This commit is contained in:
parent
68c281c590
commit
adab7335b6
|
@ -160,6 +160,14 @@ fn diversify_hash(d: [u8; 11]) -> Option<jubjub::ExtendedPoint> {
|
||||||
// exported.
|
// exported.
|
||||||
type Scalar = jubjub::Fr;
|
type Scalar = jubjub::Fr;
|
||||||
|
|
||||||
|
/// Magic human-readable strings used to identify what networks
|
||||||
|
/// Sapling Spending Keys are associated with when encoded/decoded
|
||||||
|
/// with bech32.
|
||||||
|
mod sk_hrp {
|
||||||
|
pub const MAINNET: &str = "secret-spending-key-main";
|
||||||
|
pub const TESTNET: &str = "secret-spending-key-test";
|
||||||
|
}
|
||||||
|
|
||||||
/// A _Spending Key_, as described in [protocol specification
|
/// A _Spending Key_, as described in [protocol specification
|
||||||
/// §4.2.2][ps].
|
/// §4.2.2][ps].
|
||||||
///
|
///
|
||||||
|
@ -168,7 +176,57 @@ type Scalar = jubjub::Fr;
|
||||||
///
|
///
|
||||||
/// [ps]: https://zips.z.cash/protocol/protocol.pdf#saplingkeycomponents
|
/// [ps]: https://zips.z.cash/protocol/protocol.pdf#saplingkeycomponents
|
||||||
#[derive(Copy, Clone, Debug, Eq, PartialEq)]
|
#[derive(Copy, Clone, Debug, Eq, PartialEq)]
|
||||||
pub struct SpendingKey(pub [u8; 32]);
|
pub struct SpendingKey {
|
||||||
|
network: Network,
|
||||||
|
bytes: [u8; 32],
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO: impl a From that accepts a Network?
|
||||||
|
|
||||||
|
impl From<[u8; 32]> for SpendingKey {
|
||||||
|
/// Generate a _SpendingKey_ from existing bytes.
|
||||||
|
fn from(bytes: [u8; 32]) -> Self {
|
||||||
|
Self {
|
||||||
|
network: Network::default(),
|
||||||
|
bytes,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl fmt::Display for SpendingKey {
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||||||
|
let hrp = match self.network {
|
||||||
|
Network::Mainnet => sk_hrp::MAINNET,
|
||||||
|
_ => sk_hrp::TESTNET,
|
||||||
|
};
|
||||||
|
|
||||||
|
bech32::encode_to_fmt(f, hrp, &self.bytes.to_base32()).unwrap()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl std::str::FromStr for SpendingKey {
|
||||||
|
type Err = SerializationError;
|
||||||
|
|
||||||
|
fn from_str(s: &str) -> Result<Self, Self::Err> {
|
||||||
|
match bech32::decode(s) {
|
||||||
|
Ok((hrp, bytes)) => {
|
||||||
|
let decoded = Vec::<u8>::from_base32(&bytes).unwrap();
|
||||||
|
|
||||||
|
let mut decoded_bytes = [0u8; 32];
|
||||||
|
decoded_bytes[..].copy_from_slice(&decoded[0..32]);
|
||||||
|
|
||||||
|
Ok(SpendingKey {
|
||||||
|
network: match hrp.as_str() {
|
||||||
|
ivk_hrp::MAINNET => Network::Mainnet,
|
||||||
|
_ => Network::Testnet,
|
||||||
|
},
|
||||||
|
bytes: decoded_bytes,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
Err(_) => Err(SerializationError::Parse("bech32 decoding error")),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl SpendingKey {
|
impl SpendingKey {
|
||||||
/// Generate a new _SpendingKey_.
|
/// Generate a new _SpendingKey_.
|
||||||
|
@ -183,13 +241,6 @@ impl SpendingKey {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl From<[u8; 32]> for SpendingKey {
|
|
||||||
/// Generate a _SpendingKey_ from existing bytes.
|
|
||||||
fn from(bytes: [u8; 32]) -> SpendingKey {
|
|
||||||
SpendingKey(bytes)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// A _Spend Authorizing Key_, as described in [protocol specification
|
/// A _Spend Authorizing Key_, as described in [protocol specification
|
||||||
/// §4.2.2][ps].
|
/// §4.2.2][ps].
|
||||||
///
|
///
|
||||||
|
@ -222,7 +273,7 @@ impl From<SpendingKey> for SpendAuthorizingKey {
|
||||||
/// https://zips.z.cash/protocol/protocol.pdf#saplingkeycomponents
|
/// https://zips.z.cash/protocol/protocol.pdf#saplingkeycomponents
|
||||||
/// https://zips.z.cash/protocol/protocol.pdf#concreteprfs
|
/// https://zips.z.cash/protocol/protocol.pdf#concreteprfs
|
||||||
fn from(spending_key: SpendingKey) -> SpendAuthorizingKey {
|
fn from(spending_key: SpendingKey) -> SpendAuthorizingKey {
|
||||||
let hash_bytes = prf_expand(spending_key.0, 0);
|
let hash_bytes = prf_expand(spending_key.bytes, 0);
|
||||||
|
|
||||||
Self(Scalar::from_bytes_wide(&hash_bytes))
|
Self(Scalar::from_bytes_wide(&hash_bytes))
|
||||||
}
|
}
|
||||||
|
@ -258,7 +309,7 @@ impl From<SpendingKey> for ProofAuthorizingKey {
|
||||||
/// https://zips.z.cash/protocol/protocol.pdf#saplingkeycomponents
|
/// https://zips.z.cash/protocol/protocol.pdf#saplingkeycomponents
|
||||||
/// https://zips.z.cash/protocol/protocol.pdf#concreteprfs
|
/// https://zips.z.cash/protocol/protocol.pdf#concreteprfs
|
||||||
fn from(spending_key: SpendingKey) -> ProofAuthorizingKey {
|
fn from(spending_key: SpendingKey) -> ProofAuthorizingKey {
|
||||||
let hash_bytes = prf_expand(spending_key.0, 1);
|
let hash_bytes = prf_expand(spending_key.bytes, 1);
|
||||||
|
|
||||||
Self(Scalar::from_bytes_wide(&hash_bytes))
|
Self(Scalar::from_bytes_wide(&hash_bytes))
|
||||||
}
|
}
|
||||||
|
@ -294,7 +345,7 @@ impl From<SpendingKey> for OutgoingViewingKey {
|
||||||
/// https://zips.z.cash/protocol/protocol.pdf#saplingkeycomponents
|
/// https://zips.z.cash/protocol/protocol.pdf#saplingkeycomponents
|
||||||
/// https://zips.z.cash/protocol/protocol.pdf#concreteprfs
|
/// https://zips.z.cash/protocol/protocol.pdf#concreteprfs
|
||||||
fn from(spending_key: SpendingKey) -> OutgoingViewingKey {
|
fn from(spending_key: SpendingKey) -> OutgoingViewingKey {
|
||||||
let hash_bytes = prf_expand(spending_key.0, 2);
|
let hash_bytes = prf_expand(spending_key.bytes, 2);
|
||||||
|
|
||||||
let mut bytes = [0u8; 32];
|
let mut bytes = [0u8; 32];
|
||||||
bytes[..].copy_from_slice(&hash_bytes[0..32]);
|
bytes[..].copy_from_slice(&hash_bytes[0..32]);
|
||||||
|
@ -643,6 +694,8 @@ pub struct FullViewingKey {
|
||||||
outgoing_viewing_key: OutgoingViewingKey,
|
outgoing_viewing_key: OutgoingViewingKey,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TODO: impl a From that accepts a Network?
|
||||||
|
|
||||||
impl fmt::Debug for FullViewingKey {
|
impl fmt::Debug for FullViewingKey {
|
||||||
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||||||
f.debug_struct("FullViewingKey")
|
f.debug_struct("FullViewingKey")
|
||||||
|
|
Loading…
Reference in New Issue