|
|
|
@ -1,13 +1,18 @@
|
|
|
|
|
//! Helper functions for managing light client key material.
|
|
|
|
|
use std::{
|
|
|
|
|
error,
|
|
|
|
|
fmt::{self, Display},
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
use std::{error, fmt};
|
|
|
|
|
|
|
|
|
|
use zcash_address::unified::{self, Container, Encoding, Typecode};
|
|
|
|
|
use zcash_protocol::consensus::{self, NetworkConstants};
|
|
|
|
|
use zcash_address::unified::{self, Container, Encoding, Typecode, Ufvk, Uivk};
|
|
|
|
|
use zcash_protocol::consensus;
|
|
|
|
|
use zip32::{AccountId, DiversifierIndex};
|
|
|
|
|
|
|
|
|
|
use crate::address::UnifiedAddress;
|
|
|
|
|
|
|
|
|
|
#[cfg(any(feature = "sapling", feature = "orchard"))]
|
|
|
|
|
use zcash_protocol::consensus::NetworkConstants;
|
|
|
|
|
|
|
|
|
|
#[cfg(feature = "transparent-inputs")]
|
|
|
|
|
use {
|
|
|
|
|
std::convert::TryInto,
|
|
|
|
@ -84,7 +89,6 @@ fn to_transparent_child_index(j: DiversifierIndex) -> Option<NonHardenedChildInd
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[derive(Debug)]
|
|
|
|
|
#[doc(hidden)]
|
|
|
|
|
pub enum DerivationError {
|
|
|
|
|
#[cfg(feature = "orchard")]
|
|
|
|
|
Orchard(orchard::zip32::Error),
|
|
|
|
@ -92,6 +96,21 @@ pub enum DerivationError {
|
|
|
|
|
Transparent(hdwallet::error::Error),
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl Display for DerivationError {
|
|
|
|
|
fn fmt(&self, _f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
|
|
|
|
match self {
|
|
|
|
|
#[cfg(feature = "orchard")]
|
|
|
|
|
DerivationError::Orchard(e) => write!(_f, "Orchard error: {}", e),
|
|
|
|
|
#[cfg(feature = "transparent-inputs")]
|
|
|
|
|
DerivationError::Transparent(e) => write!(_f, "Transparent error: {}", e),
|
|
|
|
|
#[cfg(not(any(feature = "orchard", feature = "transparent-inputs")))]
|
|
|
|
|
other => {
|
|
|
|
|
unreachable!("Unhandled DerivationError variant {:?}", other)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// A version identifier for the encoding of unified spending keys.
|
|
|
|
|
///
|
|
|
|
|
/// Each era corresponds to a range of block heights. During an era, the unified spending key
|
|
|
|
@ -107,28 +126,40 @@ pub enum Era {
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// A type for errors that can occur when decoding keys from their serialized representations.
|
|
|
|
|
#[cfg(feature = "unstable")]
|
|
|
|
|
#[derive(Debug, PartialEq, Eq)]
|
|
|
|
|
pub enum DecodingError {
|
|
|
|
|
#[cfg(feature = "unstable")]
|
|
|
|
|
ReadError(&'static str),
|
|
|
|
|
#[cfg(feature = "unstable")]
|
|
|
|
|
EraInvalid,
|
|
|
|
|
#[cfg(feature = "unstable")]
|
|
|
|
|
EraMismatch(Era),
|
|
|
|
|
#[cfg(feature = "unstable")]
|
|
|
|
|
TypecodeInvalid,
|
|
|
|
|
#[cfg(feature = "unstable")]
|
|
|
|
|
LengthInvalid,
|
|
|
|
|
#[cfg(feature = "unstable")]
|
|
|
|
|
LengthMismatch(Typecode, u32),
|
|
|
|
|
#[cfg(feature = "unstable")]
|
|
|
|
|
InsufficientData(Typecode),
|
|
|
|
|
/// The key data could not be decoded from its string representation to a valid key.
|
|
|
|
|
KeyDataInvalid(Typecode),
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[cfg(feature = "unstable")]
|
|
|
|
|
impl std::fmt::Display for DecodingError {
|
|
|
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
|
|
|
match self {
|
|
|
|
|
#[cfg(feature = "unstable")]
|
|
|
|
|
DecodingError::ReadError(s) => write!(f, "Read error: {}", s),
|
|
|
|
|
#[cfg(feature = "unstable")]
|
|
|
|
|
DecodingError::EraInvalid => write!(f, "Invalid era"),
|
|
|
|
|
#[cfg(feature = "unstable")]
|
|
|
|
|
DecodingError::EraMismatch(e) => write!(f, "Era mismatch: actual {:?}", e),
|
|
|
|
|
#[cfg(feature = "unstable")]
|
|
|
|
|
DecodingError::TypecodeInvalid => write!(f, "Invalid typecode"),
|
|
|
|
|
#[cfg(feature = "unstable")]
|
|
|
|
|
DecodingError::LengthInvalid => write!(f, "Invalid length"),
|
|
|
|
|
#[cfg(feature = "unstable")]
|
|
|
|
|
DecodingError::LengthMismatch(t, l) => {
|
|
|
|
|
write!(
|
|
|
|
|
f,
|
|
|
|
@ -136,10 +167,11 @@ impl std::fmt::Display for DecodingError {
|
|
|
|
|
l, t
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
#[cfg(feature = "unstable")]
|
|
|
|
|
DecodingError::InsufficientData(t) => {
|
|
|
|
|
write!(f, "Insufficient data for typecode {:?}", t)
|
|
|
|
|
}
|
|
|
|
|
DecodingError::KeyDataInvalid(t) => write!(f, "Invalid key data for typecode {:?}", t),
|
|
|
|
|
DecodingError::KeyDataInvalid(t) => write!(f, "Invalid key data for key type {:?}", t),
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
@ -186,19 +218,37 @@ impl UnifiedSpendingKey {
|
|
|
|
|
panic!("ZIP 32 seeds MUST be at least 32 bytes");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Ok(UnifiedSpendingKey {
|
|
|
|
|
UnifiedSpendingKey::from_checked_parts(
|
|
|
|
|
#[cfg(feature = "transparent-inputs")]
|
|
|
|
|
transparent: legacy::AccountPrivKey::from_seed(_params, seed, _account)
|
|
|
|
|
legacy::AccountPrivKey::from_seed(_params, seed, _account)
|
|
|
|
|
.map_err(DerivationError::Transparent)?,
|
|
|
|
|
#[cfg(feature = "sapling")]
|
|
|
|
|
sapling: sapling::spending_key(seed, _params.coin_type(), _account),
|
|
|
|
|
sapling::spending_key(seed, _params.coin_type(), _account),
|
|
|
|
|
#[cfg(feature = "orchard")]
|
|
|
|
|
orchard: orchard::keys::SpendingKey::from_zip32_seed(
|
|
|
|
|
seed,
|
|
|
|
|
_params.coin_type(),
|
|
|
|
|
_account,
|
|
|
|
|
)
|
|
|
|
|
orchard::keys::SpendingKey::from_zip32_seed(seed, _params.coin_type(), _account)
|
|
|
|
|
.map_err(DerivationError::Orchard)?,
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Construct a USK from its constituent parts, after verifying that UIVK derivation can
|
|
|
|
|
/// succeed.
|
|
|
|
|
fn from_checked_parts(
|
|
|
|
|
#[cfg(feature = "transparent-inputs")] transparent: legacy::AccountPrivKey,
|
|
|
|
|
#[cfg(feature = "sapling")] sapling: sapling::ExtendedSpendingKey,
|
|
|
|
|
#[cfg(feature = "orchard")] orchard: orchard::keys::SpendingKey,
|
|
|
|
|
) -> Result<UnifiedSpendingKey, DerivationError> {
|
|
|
|
|
// Verify that FVK and IVK derivation succeed; we don't want to construct a USK
|
|
|
|
|
// that can't derive transparent addresses.
|
|
|
|
|
#[cfg(feature = "transparent-inputs")]
|
|
|
|
|
let _ = transparent.to_account_pubkey().derive_external_ivk()?;
|
|
|
|
|
|
|
|
|
|
Ok(UnifiedSpendingKey {
|
|
|
|
|
#[cfg(feature = "transparent-inputs")]
|
|
|
|
|
transparent,
|
|
|
|
|
#[cfg(feature = "sapling")]
|
|
|
|
|
sapling,
|
|
|
|
|
#[cfg(feature = "orchard")]
|
|
|
|
|
orchard,
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
@ -388,14 +438,15 @@ impl UnifiedSpendingKey {
|
|
|
|
|
let has_transparent = true;
|
|
|
|
|
|
|
|
|
|
if has_orchard && has_sapling && has_transparent {
|
|
|
|
|
return Ok(UnifiedSpendingKey {
|
|
|
|
|
#[cfg(feature = "orchard")]
|
|
|
|
|
orchard: orchard.unwrap(),
|
|
|
|
|
#[cfg(feature = "sapling")]
|
|
|
|
|
sapling: sapling.unwrap(),
|
|
|
|
|
return UnifiedSpendingKey::from_checked_parts(
|
|
|
|
|
#[cfg(feature = "transparent-inputs")]
|
|
|
|
|
transparent: transparent.unwrap(),
|
|
|
|
|
});
|
|
|
|
|
transparent.unwrap(),
|
|
|
|
|
#[cfg(feature = "sapling")]
|
|
|
|
|
sapling.unwrap(),
|
|
|
|
|
#[cfg(feature = "orchard")]
|
|
|
|
|
orchard.unwrap(),
|
|
|
|
|
)
|
|
|
|
|
.map_err(|_| DecodingError::KeyDataInvalid(Typecode::P2pkh));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
@ -457,6 +508,7 @@ impl fmt::Display for AddressGenerationError {
|
|
|
|
|
i
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
#[cfg(feature = "sapling")]
|
|
|
|
|
AddressGenerationError::InvalidSaplingDiversifierIndex(i) => {
|
|
|
|
|
write!(
|
|
|
|
|
f,
|
|
|
|
@ -553,9 +605,15 @@ impl UnifiedAddressRequest {
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[cfg(feature = "transparent-inputs")]
|
|
|
|
|
impl From<hdwallet::error::Error> for DerivationError {
|
|
|
|
|
fn from(e: hdwallet::error::Error) -> Self {
|
|
|
|
|
DerivationError::Transparent(e)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// A [ZIP 316](https://zips.z.cash/zip-0316) unified full viewing key.
|
|
|
|
|
#[derive(Clone, Debug)]
|
|
|
|
|
#[doc(hidden)]
|
|
|
|
|
pub struct UnifiedFullViewingKey {
|
|
|
|
|
#[cfg(feature = "transparent-inputs")]
|
|
|
|
|
transparent: Option<legacy::AccountPubKey>,
|
|
|
|
@ -568,37 +626,56 @@ pub struct UnifiedFullViewingKey {
|
|
|
|
|
|
|
|
|
|
#[doc(hidden)]
|
|
|
|
|
impl UnifiedFullViewingKey {
|
|
|
|
|
/// Construct a new unified full viewing key, if the required components are present.
|
|
|
|
|
/// Construct a new unified full viewing key.
|
|
|
|
|
///
|
|
|
|
|
/// This method is only available when the `test-dependencies` feature is enabled,
|
|
|
|
|
/// as derivation from the USK or deserialization from the serialized form should
|
|
|
|
|
/// be used instead.
|
|
|
|
|
#[cfg(any(test, feature = "test-dependencies"))]
|
|
|
|
|
pub fn new(
|
|
|
|
|
#[cfg(feature = "transparent-inputs")] transparent: Option<legacy::AccountPubKey>,
|
|
|
|
|
#[cfg(feature = "sapling")] sapling: Option<sapling::DiversifiableFullViewingKey>,
|
|
|
|
|
#[cfg(feature = "orchard")] orchard: Option<orchard::keys::FullViewingKey>,
|
|
|
|
|
// TODO: Implement construction of UFVKs with metadata items.
|
|
|
|
|
) -> Option<UnifiedFullViewingKey> {
|
|
|
|
|
#[cfg(feature = "orchard")]
|
|
|
|
|
let has_orchard = orchard.is_some();
|
|
|
|
|
#[cfg(not(feature = "orchard"))]
|
|
|
|
|
let has_orchard = false;
|
|
|
|
|
#[cfg(feature = "sapling")]
|
|
|
|
|
let has_sapling = sapling.is_some();
|
|
|
|
|
#[cfg(not(feature = "sapling"))]
|
|
|
|
|
let has_sapling = false;
|
|
|
|
|
|
|
|
|
|
if has_orchard || has_sapling {
|
|
|
|
|
Some(UnifiedFullViewingKey {
|
|
|
|
|
) -> Result<UnifiedFullViewingKey, DerivationError> {
|
|
|
|
|
Self::from_checked_parts(
|
|
|
|
|
#[cfg(feature = "transparent-inputs")]
|
|
|
|
|
transparent,
|
|
|
|
|
#[cfg(feature = "sapling")]
|
|
|
|
|
sapling,
|
|
|
|
|
#[cfg(feature = "orchard")]
|
|
|
|
|
orchard,
|
|
|
|
|
// We don't allow constructing new UFVKs with unknown items, but we store
|
|
|
|
|
// We don't currently allow constructing new UFVKs with unknown items, but we store
|
|
|
|
|
// this to allow parsing such UFVKs.
|
|
|
|
|
unknown: vec![],
|
|
|
|
|
})
|
|
|
|
|
} else {
|
|
|
|
|
None
|
|
|
|
|
vec![],
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Construct a UFVK from its constituent parts, after verifying that UIVK derivation can
|
|
|
|
|
/// succeed.
|
|
|
|
|
fn from_checked_parts(
|
|
|
|
|
#[cfg(feature = "transparent-inputs")] transparent: Option<legacy::AccountPubKey>,
|
|
|
|
|
#[cfg(feature = "sapling")] sapling: Option<sapling::DiversifiableFullViewingKey>,
|
|
|
|
|
#[cfg(feature = "orchard")] orchard: Option<orchard::keys::FullViewingKey>,
|
|
|
|
|
unknown: Vec<(u32, Vec<u8>)>,
|
|
|
|
|
) -> Result<UnifiedFullViewingKey, DerivationError> {
|
|
|
|
|
// Verify that IVK derivation succeeds; we don't want to construct a UFVK
|
|
|
|
|
// that can't derive transparent addresses.
|
|
|
|
|
#[cfg(feature = "transparent-inputs")]
|
|
|
|
|
let _ = transparent
|
|
|
|
|
.as_ref()
|
|
|
|
|
.map(|t| t.derive_external_ivk())
|
|
|
|
|
.transpose()?;
|
|
|
|
|
|
|
|
|
|
Ok(UnifiedFullViewingKey {
|
|
|
|
|
#[cfg(feature = "transparent-inputs")]
|
|
|
|
|
transparent,
|
|
|
|
|
#[cfg(feature = "sapling")]
|
|
|
|
|
sapling,
|
|
|
|
|
#[cfg(feature = "orchard")]
|
|
|
|
|
orchard,
|
|
|
|
|
unknown,
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Parses a `UnifiedFullViewingKey` from its [ZIP 316] string encoding.
|
|
|
|
@ -614,6 +691,13 @@ impl UnifiedFullViewingKey {
|
|
|
|
|
));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Self::parse(&ufvk).map_err(|e| e.to_string())
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Parses a `UnifiedFullViewingKey` from its [ZIP 316] string encoding.
|
|
|
|
|
///
|
|
|
|
|
/// [ZIP 316]: https://zips.z.cash/zip-0316
|
|
|
|
|
pub fn parse(ufvk: &Ufvk) -> Result<Self, DecodingError> {
|
|
|
|
|
#[cfg(feature = "orchard")]
|
|
|
|
|
let mut orchard = None;
|
|
|
|
|
#[cfg(feature = "sapling")]
|
|
|
|
@ -629,21 +713,21 @@ impl UnifiedFullViewingKey {
|
|
|
|
|
.filter_map(|receiver| match receiver {
|
|
|
|
|
#[cfg(feature = "orchard")]
|
|
|
|
|
unified::Fvk::Orchard(data) => orchard::keys::FullViewingKey::from_bytes(data)
|
|
|
|
|
.ok_or("Invalid Orchard FVK in Unified FVK")
|
|
|
|
|
.ok_or(DecodingError::KeyDataInvalid(Typecode::Orchard))
|
|
|
|
|
.map(|addr| {
|
|
|
|
|
orchard = Some(addr);
|
|
|
|
|
None
|
|
|
|
|
})
|
|
|
|
|
.transpose(),
|
|
|
|
|
#[cfg(not(feature = "orchard"))]
|
|
|
|
|
unified::Fvk::Orchard(data) => Some(Ok::<_, &'static str>((
|
|
|
|
|
unified::Fvk::Orchard(data) => Some(Ok::<_, DecodingError>((
|
|
|
|
|
u32::from(unified::Typecode::Orchard),
|
|
|
|
|
data.to_vec(),
|
|
|
|
|
))),
|
|
|
|
|
#[cfg(feature = "sapling")]
|
|
|
|
|
unified::Fvk::Sapling(data) => {
|
|
|
|
|
sapling::DiversifiableFullViewingKey::from_bytes(data)
|
|
|
|
|
.ok_or("Invalid Sapling FVK in Unified FVK")
|
|
|
|
|
.ok_or(DecodingError::KeyDataInvalid(Typecode::Sapling))
|
|
|
|
|
.map(|pa| {
|
|
|
|
|
sapling = Some(pa);
|
|
|
|
|
None
|
|
|
|
@ -651,20 +735,20 @@ impl UnifiedFullViewingKey {
|
|
|
|
|
.transpose()
|
|
|
|
|
}
|
|
|
|
|
#[cfg(not(feature = "sapling"))]
|
|
|
|
|
unified::Fvk::Sapling(data) => Some(Ok::<_, &'static str>((
|
|
|
|
|
unified::Fvk::Sapling(data) => Some(Ok::<_, DecodingError>((
|
|
|
|
|
u32::from(unified::Typecode::Sapling),
|
|
|
|
|
data.to_vec(),
|
|
|
|
|
))),
|
|
|
|
|
#[cfg(feature = "transparent-inputs")]
|
|
|
|
|
unified::Fvk::P2pkh(data) => legacy::AccountPubKey::deserialize(data)
|
|
|
|
|
.map_err(|_| "Invalid transparent FVK in Unified FVK")
|
|
|
|
|
.map_err(|_| DecodingError::KeyDataInvalid(Typecode::P2pkh))
|
|
|
|
|
.map(|tfvk| {
|
|
|
|
|
transparent = Some(tfvk);
|
|
|
|
|
None
|
|
|
|
|
})
|
|
|
|
|
.transpose(),
|
|
|
|
|
#[cfg(not(feature = "transparent-inputs"))]
|
|
|
|
|
unified::Fvk::P2pkh(data) => Some(Ok::<_, &'static str>((
|
|
|
|
|
unified::Fvk::P2pkh(data) => Some(Ok::<_, DecodingError>((
|
|
|
|
|
u32::from(unified::Typecode::P2pkh),
|
|
|
|
|
data.to_vec(),
|
|
|
|
|
))),
|
|
|
|
@ -672,7 +756,7 @@ impl UnifiedFullViewingKey {
|
|
|
|
|
})
|
|
|
|
|
.collect::<Result<_, _>>()?;
|
|
|
|
|
|
|
|
|
|
Ok(Self {
|
|
|
|
|
Self::from_checked_parts(
|
|
|
|
|
#[cfg(feature = "transparent-inputs")]
|
|
|
|
|
transparent,
|
|
|
|
|
#[cfg(feature = "sapling")]
|
|
|
|
@ -680,11 +764,17 @@ impl UnifiedFullViewingKey {
|
|
|
|
|
#[cfg(feature = "orchard")]
|
|
|
|
|
orchard,
|
|
|
|
|
unknown,
|
|
|
|
|
})
|
|
|
|
|
)
|
|
|
|
|
.map_err(|_| DecodingError::KeyDataInvalid(Typecode::P2pkh))
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Returns the string encoding of this `UnifiedFullViewingKey` for the given network.
|
|
|
|
|
pub fn encode<P: consensus::Parameters>(&self, params: &P) -> String {
|
|
|
|
|
self.to_ufvk().encode(¶ms.network_type())
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Returns the string encoding of this `UnifiedFullViewingKey` for the given network.
|
|
|
|
|
fn to_ufvk(&self) -> Ufvk {
|
|
|
|
|
let items = std::iter::empty().chain(self.unknown.iter().map(|(typecode, data)| {
|
|
|
|
|
unified::Fvk::Unknown {
|
|
|
|
|
typecode: *typecode,
|
|
|
|
@ -713,9 +803,24 @@ impl UnifiedFullViewingKey {
|
|
|
|
|
.map(unified::Fvk::P2pkh),
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
let ufvk = unified::Ufvk::try_from_items(items.collect())
|
|
|
|
|
.expect("UnifiedFullViewingKey should only be constructed safely");
|
|
|
|
|
ufvk.encode(¶ms.network_type())
|
|
|
|
|
unified::Ufvk::try_from_items(items.collect())
|
|
|
|
|
.expect("UnifiedFullViewingKey should only be constructed safely")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Derives a Unified Incoming Viewing Key from this Unified Full Viewing Key.
|
|
|
|
|
pub fn to_unified_incoming_viewing_key(&self) -> UnifiedIncomingViewingKey {
|
|
|
|
|
UnifiedIncomingViewingKey {
|
|
|
|
|
#[cfg(feature = "transparent-inputs")]
|
|
|
|
|
transparent: self.transparent.as_ref().map(|t| {
|
|
|
|
|
t.derive_external_ivk()
|
|
|
|
|
.expect("Transparent IVK derivation was checked at construction.")
|
|
|
|
|
}),
|
|
|
|
|
#[cfg(feature = "sapling")]
|
|
|
|
|
sapling: self.sapling.as_ref().map(|s| s.to_external_ivk()),
|
|
|
|
|
#[cfg(feature = "orchard")]
|
|
|
|
|
orchard: self.orchard.as_ref().map(|o| o.to_ivk(Scope::External)),
|
|
|
|
|
unknown: Vec::new(),
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Returns the transparent component of the unified key at the
|
|
|
|
@ -737,6 +842,229 @@ impl UnifiedFullViewingKey {
|
|
|
|
|
self.orchard.as_ref()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Attempts to derive the Unified Address for the given diversifier index and
|
|
|
|
|
/// receiver types.
|
|
|
|
|
///
|
|
|
|
|
/// Returns `None` if the specified index does not produce a valid diversifier.
|
|
|
|
|
pub fn address(
|
|
|
|
|
&self,
|
|
|
|
|
j: DiversifierIndex,
|
|
|
|
|
request: UnifiedAddressRequest,
|
|
|
|
|
) -> Result<UnifiedAddress, AddressGenerationError> {
|
|
|
|
|
self.to_unified_incoming_viewing_key().address(j, request)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Searches the diversifier space starting at diversifier index `j` for one which will
|
|
|
|
|
/// produce a valid diversifier, and return the Unified Address constructed using that
|
|
|
|
|
/// diversifier along with the index at which the valid diversifier was found.
|
|
|
|
|
///
|
|
|
|
|
/// Returns an `Err(AddressGenerationError)` if no valid diversifier exists or if the features
|
|
|
|
|
/// required to satisfy the unified address request are not properly enabled.
|
|
|
|
|
#[allow(unused_mut)]
|
|
|
|
|
pub fn find_address(
|
|
|
|
|
&self,
|
|
|
|
|
mut j: DiversifierIndex,
|
|
|
|
|
request: UnifiedAddressRequest,
|
|
|
|
|
) -> Result<(UnifiedAddress, DiversifierIndex), AddressGenerationError> {
|
|
|
|
|
self.to_unified_incoming_viewing_key()
|
|
|
|
|
.find_address(j, request)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Find the Unified Address corresponding to the smallest valid diversifier index, along with
|
|
|
|
|
/// that index.
|
|
|
|
|
///
|
|
|
|
|
/// Returns an `Err(AddressGenerationError)` if no valid diversifier exists or if the features
|
|
|
|
|
/// required to satisfy the unified address request are not properly enabled.
|
|
|
|
|
pub fn default_address(
|
|
|
|
|
&self,
|
|
|
|
|
request: UnifiedAddressRequest,
|
|
|
|
|
) -> Result<(UnifiedAddress, DiversifierIndex), AddressGenerationError> {
|
|
|
|
|
self.find_address(DiversifierIndex::new(), request)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// A [ZIP 316](https://zips.z.cash/zip-0316) unified incoming viewing key.
|
|
|
|
|
#[derive(Clone, Debug)]
|
|
|
|
|
pub struct UnifiedIncomingViewingKey {
|
|
|
|
|
#[cfg(feature = "transparent-inputs")]
|
|
|
|
|
transparent: Option<zcash_primitives::legacy::keys::ExternalIvk>,
|
|
|
|
|
#[cfg(feature = "sapling")]
|
|
|
|
|
sapling: Option<::sapling::zip32::IncomingViewingKey>,
|
|
|
|
|
#[cfg(feature = "orchard")]
|
|
|
|
|
orchard: Option<orchard::keys::IncomingViewingKey>,
|
|
|
|
|
/// Stores the unrecognized elements of the unified encoding.
|
|
|
|
|
unknown: Vec<(u32, Vec<u8>)>,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl UnifiedIncomingViewingKey {
|
|
|
|
|
/// Construct a new unified incoming viewing key.
|
|
|
|
|
///
|
|
|
|
|
/// This method is only available when the `test-dependencies` feature is enabled,
|
|
|
|
|
/// as derivation from the UFVK or deserialization from the serialized form should
|
|
|
|
|
/// be used instead.
|
|
|
|
|
#[cfg(any(test, feature = "test-dependencies"))]
|
|
|
|
|
pub fn new(
|
|
|
|
|
#[cfg(feature = "transparent-inputs")] transparent: Option<
|
|
|
|
|
zcash_primitives::legacy::keys::ExternalIvk,
|
|
|
|
|
>,
|
|
|
|
|
#[cfg(feature = "sapling")] sapling: Option<::sapling::zip32::IncomingViewingKey>,
|
|
|
|
|
#[cfg(feature = "orchard")] orchard: Option<orchard::keys::IncomingViewingKey>,
|
|
|
|
|
// TODO: Implement construction of UIVKs with metadata items.
|
|
|
|
|
) -> UnifiedIncomingViewingKey {
|
|
|
|
|
UnifiedIncomingViewingKey {
|
|
|
|
|
#[cfg(feature = "transparent-inputs")]
|
|
|
|
|
transparent,
|
|
|
|
|
#[cfg(feature = "sapling")]
|
|
|
|
|
sapling,
|
|
|
|
|
#[cfg(feature = "orchard")]
|
|
|
|
|
orchard,
|
|
|
|
|
// We don't allow constructing new UFVKs with unknown items, but we store
|
|
|
|
|
// this to allow parsing such UFVKs.
|
|
|
|
|
unknown: vec![],
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Parses a `UnifiedFullViewingKey` from its [ZIP 316] string encoding.
|
|
|
|
|
///
|
|
|
|
|
/// [ZIP 316]: https://zips.z.cash/zip-0316
|
|
|
|
|
pub fn decode<P: consensus::Parameters>(params: &P, encoding: &str) -> Result<Self, String> {
|
|
|
|
|
let (net, ufvk) = unified::Uivk::decode(encoding).map_err(|e| e.to_string())?;
|
|
|
|
|
let expected_net = params.network_type();
|
|
|
|
|
if net != expected_net {
|
|
|
|
|
return Err(format!(
|
|
|
|
|
"UIVK is for network {:?} but we expected {:?}",
|
|
|
|
|
net, expected_net,
|
|
|
|
|
));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Self::parse(&ufvk).map_err(|e| e.to_string())
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Constructs a unified incoming viewing key from a parsed unified encoding.
|
|
|
|
|
fn parse(uivk: &Uivk) -> Result<Self, DecodingError> {
|
|
|
|
|
#[cfg(feature = "orchard")]
|
|
|
|
|
let mut orchard = None;
|
|
|
|
|
#[cfg(feature = "sapling")]
|
|
|
|
|
let mut sapling = None;
|
|
|
|
|
#[cfg(feature = "transparent-inputs")]
|
|
|
|
|
let mut transparent = None;
|
|
|
|
|
|
|
|
|
|
let mut unknown = vec![];
|
|
|
|
|
|
|
|
|
|
// We can use as-parsed order here for efficiency, because we're breaking out the
|
|
|
|
|
// receivers we support from the unknown receivers.
|
|
|
|
|
for receiver in uivk.items_as_parsed() {
|
|
|
|
|
match receiver {
|
|
|
|
|
unified::Ivk::Orchard(data) => {
|
|
|
|
|
#[cfg(feature = "orchard")]
|
|
|
|
|
{
|
|
|
|
|
orchard = Some(
|
|
|
|
|
Option::from(orchard::keys::IncomingViewingKey::from_bytes(data))
|
|
|
|
|
.ok_or(DecodingError::KeyDataInvalid(Typecode::Orchard))?,
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[cfg(not(feature = "orchard"))]
|
|
|
|
|
unknown.push((u32::from(unified::Typecode::Orchard), data.to_vec()));
|
|
|
|
|
}
|
|
|
|
|
unified::Ivk::Sapling(data) => {
|
|
|
|
|
#[cfg(feature = "sapling")]
|
|
|
|
|
{
|
|
|
|
|
sapling = Some(
|
|
|
|
|
Option::from(::sapling::zip32::IncomingViewingKey::from_bytes(data))
|
|
|
|
|
.ok_or(DecodingError::KeyDataInvalid(Typecode::Sapling))?,
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[cfg(not(feature = "sapling"))]
|
|
|
|
|
unknown.push((u32::from(unified::Typecode::Sapling), data.to_vec()));
|
|
|
|
|
}
|
|
|
|
|
unified::Ivk::P2pkh(data) => {
|
|
|
|
|
#[cfg(feature = "transparent-inputs")]
|
|
|
|
|
{
|
|
|
|
|
transparent = Some(
|
|
|
|
|
legacy::ExternalIvk::deserialize(data)
|
|
|
|
|
.map_err(|_| DecodingError::KeyDataInvalid(Typecode::P2pkh))?,
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[cfg(not(feature = "transparent-inputs"))]
|
|
|
|
|
unknown.push((u32::from(unified::Typecode::P2pkh), data.to_vec()));
|
|
|
|
|
}
|
|
|
|
|
unified::Ivk::Unknown { typecode, data } => {
|
|
|
|
|
unknown.push((*typecode, data.clone()));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Ok(Self {
|
|
|
|
|
#[cfg(feature = "transparent-inputs")]
|
|
|
|
|
transparent,
|
|
|
|
|
#[cfg(feature = "sapling")]
|
|
|
|
|
sapling,
|
|
|
|
|
#[cfg(feature = "orchard")]
|
|
|
|
|
orchard,
|
|
|
|
|
unknown,
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Returns the string encoding of this `UnifiedFullViewingKey` for the given network.
|
|
|
|
|
pub fn encode<P: consensus::Parameters>(&self, params: &P) -> String {
|
|
|
|
|
self.render().encode(¶ms.network_type())
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Converts this unified incoming viewing key to a unified encoding.
|
|
|
|
|
fn render(&self) -> Uivk {
|
|
|
|
|
let items = std::iter::empty().chain(self.unknown.iter().map(|(typecode, data)| {
|
|
|
|
|
unified::Ivk::Unknown {
|
|
|
|
|
typecode: *typecode,
|
|
|
|
|
data: data.clone(),
|
|
|
|
|
}
|
|
|
|
|
}));
|
|
|
|
|
#[cfg(feature = "orchard")]
|
|
|
|
|
let items = items.chain(
|
|
|
|
|
self.orchard
|
|
|
|
|
.as_ref()
|
|
|
|
|
.map(|ivk| ivk.to_bytes())
|
|
|
|
|
.map(unified::Ivk::Orchard),
|
|
|
|
|
);
|
|
|
|
|
#[cfg(feature = "sapling")]
|
|
|
|
|
let items = items.chain(
|
|
|
|
|
self.sapling
|
|
|
|
|
.as_ref()
|
|
|
|
|
.map(|divk| divk.to_bytes())
|
|
|
|
|
.map(unified::Ivk::Sapling),
|
|
|
|
|
);
|
|
|
|
|
#[cfg(feature = "transparent-inputs")]
|
|
|
|
|
let items = items.chain(
|
|
|
|
|
self.transparent
|
|
|
|
|
.as_ref()
|
|
|
|
|
.map(|tivk| tivk.serialize().try_into().unwrap())
|
|
|
|
|
.map(unified::Ivk::P2pkh),
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
unified::Uivk::try_from_items(items.collect())
|
|
|
|
|
.expect("UnifiedIncomingViewingKey should only be constructed safely.")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Returns the Transparent external IVK, if present.
|
|
|
|
|
#[cfg(feature = "transparent-inputs")]
|
|
|
|
|
pub fn transparent(&self) -> &Option<zcash_primitives::legacy::keys::ExternalIvk> {
|
|
|
|
|
&self.transparent
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Returns the Sapling IVK, if present.
|
|
|
|
|
#[cfg(feature = "sapling")]
|
|
|
|
|
pub fn sapling(&self) -> &Option<::sapling::zip32::IncomingViewingKey> {
|
|
|
|
|
&self.sapling
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Returns the Orchard IVK, if present.
|
|
|
|
|
#[cfg(feature = "orchard")]
|
|
|
|
|
pub fn orchard(&self) -> &Option<orchard::keys::IncomingViewingKey> {
|
|
|
|
|
&self.orchard
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Attempts to derive the Unified Address for the given diversifier index and
|
|
|
|
|
/// receiver types.
|
|
|
|
|
///
|
|
|
|
@ -744,20 +1072,20 @@ impl UnifiedFullViewingKey {
|
|
|
|
|
pub fn address(
|
|
|
|
|
&self,
|
|
|
|
|
_j: DiversifierIndex,
|
|
|
|
|
_request: UnifiedAddressRequest,
|
|
|
|
|
request: UnifiedAddressRequest,
|
|
|
|
|
) -> Result<UnifiedAddress, AddressGenerationError> {
|
|
|
|
|
#[cfg(feature = "orchard")]
|
|
|
|
|
let mut orchard = None;
|
|
|
|
|
if _request.has_orchard {
|
|
|
|
|
if request.has_orchard {
|
|
|
|
|
#[cfg(not(feature = "orchard"))]
|
|
|
|
|
return Err(AddressGenerationError::ReceiverTypeNotSupported(
|
|
|
|
|
Typecode::Orchard,
|
|
|
|
|
));
|
|
|
|
|
|
|
|
|
|
#[cfg(feature = "orchard")]
|
|
|
|
|
if let Some(ofvk) = &self.orchard {
|
|
|
|
|
if let Some(oivk) = &self.orchard {
|
|
|
|
|
let orchard_j = orchard::keys::DiversifierIndex::from(*_j.as_bytes());
|
|
|
|
|
orchard = Some(ofvk.address_at(orchard_j, Scope::External))
|
|
|
|
|
orchard = Some(oivk.address_at(orchard_j))
|
|
|
|
|
} else {
|
|
|
|
|
return Err(AddressGenerationError::KeyNotAvailable(Typecode::Orchard));
|
|
|
|
|
}
|
|
|
|
@ -765,20 +1093,19 @@ impl UnifiedFullViewingKey {
|
|
|
|
|
|
|
|
|
|
#[cfg(feature = "sapling")]
|
|
|
|
|
let mut sapling = None;
|
|
|
|
|
if _request.has_sapling {
|
|
|
|
|
if request.has_sapling {
|
|
|
|
|
#[cfg(not(feature = "sapling"))]
|
|
|
|
|
return Err(AddressGenerationError::ReceiverTypeNotSupported(
|
|
|
|
|
Typecode::Sapling,
|
|
|
|
|
));
|
|
|
|
|
|
|
|
|
|
#[cfg(feature = "sapling")]
|
|
|
|
|
if let Some(extfvk) = &self.sapling {
|
|
|
|
|
if let Some(divk) = &self.sapling {
|
|
|
|
|
// If a Sapling receiver type is requested, we must be able to construct an
|
|
|
|
|
// address; if we're unable to do so, then no Unified Address exists at this
|
|
|
|
|
// diversifier and we use `?` to early-return from this method.
|
|
|
|
|
sapling = Some(
|
|
|
|
|
extfvk
|
|
|
|
|
.address(_j)
|
|
|
|
|
divk.address_at(_j)
|
|
|
|
|
.ok_or(AddressGenerationError::InvalidSaplingDiversifierIndex(_j))?,
|
|
|
|
|
);
|
|
|
|
|
} else {
|
|
|
|
@ -788,14 +1115,14 @@ impl UnifiedFullViewingKey {
|
|
|
|
|
|
|
|
|
|
#[cfg(feature = "transparent-inputs")]
|
|
|
|
|
let mut transparent = None;
|
|
|
|
|
if _request.has_p2pkh {
|
|
|
|
|
if request.has_p2pkh {
|
|
|
|
|
#[cfg(not(feature = "transparent-inputs"))]
|
|
|
|
|
return Err(AddressGenerationError::ReceiverTypeNotSupported(
|
|
|
|
|
Typecode::P2pkh,
|
|
|
|
|
));
|
|
|
|
|
|
|
|
|
|
#[cfg(feature = "transparent-inputs")]
|
|
|
|
|
if let Some(tfvk) = self.transparent.as_ref() {
|
|
|
|
|
if let Some(tivk) = self.transparent.as_ref() {
|
|
|
|
|
// If a transparent receiver type is requested, we must be able to construct an
|
|
|
|
|
// address; if we're unable to do so, then no Unified Address exists at this
|
|
|
|
|
// diversifier.
|
|
|
|
@ -803,8 +1130,7 @@ impl UnifiedFullViewingKey {
|
|
|
|
|
.ok_or(AddressGenerationError::InvalidTransparentChildIndex(_j))?;
|
|
|
|
|
|
|
|
|
|
transparent = Some(
|
|
|
|
|
tfvk.derive_external_ivk()
|
|
|
|
|
.and_then(|tivk| tivk.derive_address(transparent_j))
|
|
|
|
|
tivk.derive_address(transparent_j)
|
|
|
|
|
.map_err(|_| AddressGenerationError::InvalidTransparentChildIndex(_j))?,
|
|
|
|
|
);
|
|
|
|
|
} else {
|
|
|
|
@ -905,16 +1231,20 @@ pub mod testing {
|
|
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
|
mod tests {
|
|
|
|
|
use super::UnifiedFullViewingKey;
|
|
|
|
|
|
|
|
|
|
use proptest::prelude::proptest;
|
|
|
|
|
|
|
|
|
|
#[cfg(any(
|
|
|
|
|
feature = "orchard",
|
|
|
|
|
feature = "sapling",
|
|
|
|
|
feature = "transparent-inputs"
|
|
|
|
|
))]
|
|
|
|
|
use {zcash_primitives::consensus::MAIN_NETWORK, zip32::AccountId};
|
|
|
|
|
|
|
|
|
|
#[cfg(any(feature = "sapling", feature = "orchard"))]
|
|
|
|
|
use {
|
|
|
|
|
super::{UnifiedFullViewingKey, UnifiedIncomingViewingKey},
|
|
|
|
|
zcash_address::unified::{Encoding, Uivk},
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
#[cfg(feature = "orchard")]
|
|
|
|
|
use zip32::Scope;
|
|
|
|
|
|
|
|
|
|
#[cfg(feature = "sapling")]
|
|
|
|
|
use super::sapling;
|
|
|
|
|
|
|
|
|
@ -922,18 +1252,12 @@ mod tests {
|
|
|
|
|
use {
|
|
|
|
|
crate::{address::Address, encoding::AddressCodec},
|
|
|
|
|
zcash_address::test_vectors,
|
|
|
|
|
zcash_primitives::legacy::{
|
|
|
|
|
self,
|
|
|
|
|
keys::{AccountPrivKey, IncomingViewingKey},
|
|
|
|
|
},
|
|
|
|
|
zcash_primitives::legacy::keys::{AccountPrivKey, IncomingViewingKey},
|
|
|
|
|
zip32::DiversifierIndex,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
#[cfg(feature = "unstable")]
|
|
|
|
|
use {
|
|
|
|
|
super::{testing::arb_unified_spending_key, Era, UnifiedSpendingKey},
|
|
|
|
|
zcash_primitives::consensus::Network,
|
|
|
|
|
};
|
|
|
|
|
use super::{testing::arb_unified_spending_key, Era, UnifiedSpendingKey};
|
|
|
|
|
|
|
|
|
|
#[cfg(all(feature = "orchard", feature = "unstable"))]
|
|
|
|
|
use subtle::ConstantTimeEq;
|
|
|
|
@ -956,8 +1280,7 @@ mod tests {
|
|
|
|
|
fn pk_to_taddr() {
|
|
|
|
|
use zcash_primitives::legacy::keys::NonHardenedChildIndex;
|
|
|
|
|
|
|
|
|
|
let taddr =
|
|
|
|
|
legacy::keys::AccountPrivKey::from_seed(&MAIN_NETWORK, &seed(), AccountId::ZERO)
|
|
|
|
|
let taddr = AccountPrivKey::from_seed(&MAIN_NETWORK, &seed(), AccountId::ZERO)
|
|
|
|
|
.unwrap()
|
|
|
|
|
.to_account_pubkey()
|
|
|
|
|
.derive_external_ivk()
|
|
|
|
@ -969,6 +1292,7 @@ mod tests {
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
#[cfg(any(feature = "orchard", feature = "sapling"))]
|
|
|
|
|
fn ufvk_round_trip() {
|
|
|
|
|
#[cfg(feature = "orchard")]
|
|
|
|
|
let orchard = {
|
|
|
|
@ -999,11 +1323,6 @@ mod tests {
|
|
|
|
|
orchard,
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
#[cfg(not(any(feature = "orchard", feature = "sapling")))]
|
|
|
|
|
assert!(ufvk.is_none());
|
|
|
|
|
|
|
|
|
|
#[cfg(any(feature = "orchard", feature = "sapling"))]
|
|
|
|
|
{
|
|
|
|
|
let ufvk = ufvk.expect("Orchard or Sapling fvk is present.");
|
|
|
|
|
let encoded = ufvk.encode(&MAIN_NETWORK);
|
|
|
|
|
|
|
|
|
@ -1043,8 +1362,7 @@ mod tests {
|
|
|
|
|
ufvk.orchard.map(|o| o.to_bytes()),
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
let decoded_with_t =
|
|
|
|
|
UnifiedFullViewingKey::decode(&MAIN_NETWORK, encoded_with_t).unwrap();
|
|
|
|
|
let decoded_with_t = UnifiedFullViewingKey::decode(&MAIN_NETWORK, encoded_with_t).unwrap();
|
|
|
|
|
#[cfg(feature = "transparent-inputs")]
|
|
|
|
|
assert_eq!(
|
|
|
|
|
decoded_with_t.transparent.map(|t| t.serialize()),
|
|
|
|
@ -1093,7 +1411,6 @@ mod tests {
|
|
|
|
|
))]
|
|
|
|
|
assert_eq!(decoded_with_t.unknown.len(), 2);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
#[cfg(feature = "transparent-inputs")]
|
|
|
|
@ -1151,10 +1468,191 @@ mod tests {
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
#[cfg(any(feature = "orchard", feature = "sapling"))]
|
|
|
|
|
fn uivk_round_trip() {
|
|
|
|
|
use zcash_primitives::consensus::NetworkType;
|
|
|
|
|
|
|
|
|
|
#[cfg(feature = "orchard")]
|
|
|
|
|
let orchard = {
|
|
|
|
|
let sk =
|
|
|
|
|
orchard::keys::SpendingKey::from_zip32_seed(&[0; 32], 0, AccountId::ZERO).unwrap();
|
|
|
|
|
Some(orchard::keys::FullViewingKey::from(&sk).to_ivk(Scope::External))
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
#[cfg(feature = "sapling")]
|
|
|
|
|
let sapling = {
|
|
|
|
|
let extsk = sapling::spending_key(&[0; 32], 0, AccountId::ZERO);
|
|
|
|
|
Some(extsk.to_diversifiable_full_viewing_key().to_external_ivk())
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
#[cfg(feature = "transparent-inputs")]
|
|
|
|
|
let transparent = {
|
|
|
|
|
let privkey =
|
|
|
|
|
AccountPrivKey::from_seed(&MAIN_NETWORK, &[0; 32], AccountId::ZERO).unwrap();
|
|
|
|
|
Some(privkey.to_account_pubkey().derive_external_ivk().unwrap())
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
let uivk = UnifiedIncomingViewingKey::new(
|
|
|
|
|
#[cfg(feature = "transparent-inputs")]
|
|
|
|
|
transparent,
|
|
|
|
|
#[cfg(feature = "sapling")]
|
|
|
|
|
sapling,
|
|
|
|
|
#[cfg(feature = "orchard")]
|
|
|
|
|
orchard,
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
let encoded = uivk.render().encode(&NetworkType::Main);
|
|
|
|
|
|
|
|
|
|
// Test encoded form against known values; these test vectors contain Orchard receivers
|
|
|
|
|
// that will be treated as unknown if the `orchard` feature is not enabled.
|
|
|
|
|
let encoded_with_t = "uivk1z28yg638vjwusmf0zc9ad2j0mpv6s42wc5kqt004aaqfu5xxxgu7mdcydn9qf723fnryt34s6jyxyw0jt7spq04c3v9ze6qe9gjjc5aglz8zv5pqtw58czd0actynww5n85z3052kzgy6cu0fyjafyp4sr4kppyrrwhwev2rr0awq6m8d66esvk6fgacggqnswg5g9gkv6t6fj9ajhyd0gmel4yzscprpzduncc0e2lywufup6fvzf6y8cefez2r99pgge5yyfuus0r60khgu895pln5e7nn77q6s9kh2uwf6lrfu06ma2kd7r05jjvl4hn6nupge8fajh0cazd7mkmz23t79w";
|
|
|
|
|
let _encoded_no_t = "uivk1020vq9j5zeqxh303sxa0zv2hn9wm9fev8x0p8yqxdwyzde9r4c90fcglc63usj0ycl2scy8zxuhtser0qrq356xfy8x3vyuxu7f6gas75svl9v9m3ctuazsu0ar8e8crtx7x6zgh4kw8xm3q4rlkpm9er2wefxhhf9pn547gpuz9vw27gsdp6c03nwlrxgzhr2g6xek0x8l5avrx9ue9lf032tr7kmhqf3nfdxg7ldfgx6yf09g";
|
|
|
|
|
|
|
|
|
|
// We test the full roundtrip only with the `sapling` and `orchard` features enabled,
|
|
|
|
|
// because we will not generate these parts of the encoding if the UIVK does not have an
|
|
|
|
|
// these parts.
|
|
|
|
|
#[cfg(all(feature = "sapling", feature = "orchard"))]
|
|
|
|
|
{
|
|
|
|
|
#[cfg(feature = "transparent-inputs")]
|
|
|
|
|
assert_eq!(encoded, encoded_with_t);
|
|
|
|
|
#[cfg(not(feature = "transparent-inputs"))]
|
|
|
|
|
assert_eq!(encoded, _encoded_no_t);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
let decoded = UnifiedIncomingViewingKey::parse(&Uivk::decode(&encoded).unwrap().1).unwrap();
|
|
|
|
|
let reencoded = decoded.render().encode(&NetworkType::Main);
|
|
|
|
|
assert_eq!(encoded, reencoded);
|
|
|
|
|
|
|
|
|
|
#[cfg(feature = "transparent-inputs")]
|
|
|
|
|
assert_eq!(
|
|
|
|
|
decoded.transparent.map(|t| t.serialize()),
|
|
|
|
|
uivk.transparent.as_ref().map(|t| t.serialize()),
|
|
|
|
|
);
|
|
|
|
|
#[cfg(feature = "sapling")]
|
|
|
|
|
assert_eq!(
|
|
|
|
|
decoded.sapling.map(|s| s.to_bytes()),
|
|
|
|
|
uivk.sapling.map(|s| s.to_bytes()),
|
|
|
|
|
);
|
|
|
|
|
#[cfg(feature = "orchard")]
|
|
|
|
|
assert_eq!(
|
|
|
|
|
decoded.orchard.map(|o| o.to_bytes()),
|
|
|
|
|
uivk.orchard.map(|o| o.to_bytes()),
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
let decoded_with_t =
|
|
|
|
|
UnifiedIncomingViewingKey::parse(&Uivk::decode(encoded_with_t).unwrap().1).unwrap();
|
|
|
|
|
#[cfg(feature = "transparent-inputs")]
|
|
|
|
|
assert_eq!(
|
|
|
|
|
decoded_with_t.transparent.map(|t| t.serialize()),
|
|
|
|
|
uivk.transparent.as_ref().map(|t| t.serialize()),
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Both Orchard and Sapling enabled
|
|
|
|
|
#[cfg(all(
|
|
|
|
|
feature = "orchard",
|
|
|
|
|
feature = "sapling",
|
|
|
|
|
feature = "transparent-inputs"
|
|
|
|
|
))]
|
|
|
|
|
assert_eq!(decoded_with_t.unknown.len(), 0);
|
|
|
|
|
#[cfg(all(
|
|
|
|
|
feature = "orchard",
|
|
|
|
|
feature = "sapling",
|
|
|
|
|
not(feature = "transparent-inputs")
|
|
|
|
|
))]
|
|
|
|
|
assert_eq!(decoded_with_t.unknown.len(), 1);
|
|
|
|
|
|
|
|
|
|
// Orchard enabled
|
|
|
|
|
#[cfg(all(
|
|
|
|
|
feature = "orchard",
|
|
|
|
|
not(feature = "sapling"),
|
|
|
|
|
feature = "transparent-inputs"
|
|
|
|
|
))]
|
|
|
|
|
assert_eq!(decoded_with_t.unknown.len(), 1);
|
|
|
|
|
#[cfg(all(
|
|
|
|
|
feature = "orchard",
|
|
|
|
|
not(feature = "sapling"),
|
|
|
|
|
not(feature = "transparent-inputs")
|
|
|
|
|
))]
|
|
|
|
|
assert_eq!(decoded_with_t.unknown.len(), 2);
|
|
|
|
|
|
|
|
|
|
// Sapling enabled
|
|
|
|
|
#[cfg(all(
|
|
|
|
|
not(feature = "orchard"),
|
|
|
|
|
feature = "sapling",
|
|
|
|
|
feature = "transparent-inputs"
|
|
|
|
|
))]
|
|
|
|
|
assert_eq!(decoded_with_t.unknown.len(), 1);
|
|
|
|
|
#[cfg(all(
|
|
|
|
|
not(feature = "orchard"),
|
|
|
|
|
feature = "sapling",
|
|
|
|
|
not(feature = "transparent-inputs")
|
|
|
|
|
))]
|
|
|
|
|
assert_eq!(decoded_with_t.unknown.len(), 2);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
#[cfg(feature = "transparent-inputs")]
|
|
|
|
|
fn uivk_derivation() {
|
|
|
|
|
use crate::keys::UnifiedAddressRequest;
|
|
|
|
|
|
|
|
|
|
use super::UnifiedSpendingKey;
|
|
|
|
|
|
|
|
|
|
for tv in test_vectors::UNIFIED {
|
|
|
|
|
let usk = UnifiedSpendingKey::from_seed(
|
|
|
|
|
&MAIN_NETWORK,
|
|
|
|
|
&tv.root_seed,
|
|
|
|
|
AccountId::try_from(tv.account).unwrap(),
|
|
|
|
|
)
|
|
|
|
|
.expect("seed produced a valid unified spending key");
|
|
|
|
|
|
|
|
|
|
let d_idx = DiversifierIndex::from(tv.diversifier_index);
|
|
|
|
|
let uivk = usk
|
|
|
|
|
.to_unified_full_viewing_key()
|
|
|
|
|
.to_unified_incoming_viewing_key();
|
|
|
|
|
|
|
|
|
|
// The test vectors contain some diversifier indices that do not generate
|
|
|
|
|
// valid Sapling addresses, so skip those.
|
|
|
|
|
#[cfg(feature = "sapling")]
|
|
|
|
|
if uivk.sapling().as_ref().unwrap().address_at(d_idx).is_none() {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
let ua = uivk
|
|
|
|
|
.address(d_idx, UnifiedAddressRequest::unsafe_new(false, true, true))
|
|
|
|
|
.unwrap_or_else(|err| {
|
|
|
|
|
panic!(
|
|
|
|
|
"unified address generation failed for account {}: {:?}",
|
|
|
|
|
tv.account, err
|
|
|
|
|
)
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
match Address::decode(&MAIN_NETWORK, tv.unified_addr) {
|
|
|
|
|
Some(Address::Unified(tvua)) => {
|
|
|
|
|
// We always derive transparent and Sapling receivers, but not
|
|
|
|
|
// every value in the test vectors has these present.
|
|
|
|
|
if tvua.transparent().is_some() {
|
|
|
|
|
assert_eq!(tvua.transparent(), ua.transparent());
|
|
|
|
|
}
|
|
|
|
|
#[cfg(feature = "sapling")]
|
|
|
|
|
if tvua.sapling().is_some() {
|
|
|
|
|
assert_eq!(tvua.sapling(), ua.sapling());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
_other => {
|
|
|
|
|
panic!(
|
|
|
|
|
"{} did not decode to a valid unified address",
|
|
|
|
|
tv.unified_addr
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
proptest! {
|
|
|
|
|
#[test]
|
|
|
|
|
#[cfg(feature = "unstable")]
|
|
|
|
|
fn prop_usk_roundtrip(usk in arb_unified_spending_key(Network::MainNetwork)) {
|
|
|
|
|
fn prop_usk_roundtrip(usk in arb_unified_spending_key(zcash_protocol::consensus::Network::MainNetwork)) {
|
|
|
|
|
let encoded = usk.to_bytes(Era::Orchard);
|
|
|
|
|
|
|
|
|
|
let encoded_len = {
|
|
|
|
|