zcash_client_backend: Generate Orchard component for USKs

This ensures that the resulting UFVKs and UAs also contain Orchard
components.
This commit is contained in:
Jack Grigg 2022-08-24 14:56:58 +00:00
parent 5dceb93ec8
commit b42344d491
1 changed files with 19 additions and 3 deletions

View File

@ -69,6 +69,7 @@ fn to_transparent_child_index(j: DiversifierIndex) -> Option<u32> {
#[derive(Debug)] #[derive(Debug)]
#[doc(hidden)] #[doc(hidden)]
pub enum DerivationError { pub enum DerivationError {
Orchard(orchard::zip32::Error),
#[cfg(feature = "transparent-inputs")] #[cfg(feature = "transparent-inputs")]
Transparent(hdwallet::error::Error), Transparent(hdwallet::error::Error),
} }
@ -82,6 +83,7 @@ pub struct UnifiedSpendingKey {
#[cfg(feature = "transparent-inputs")] #[cfg(feature = "transparent-inputs")]
transparent: legacy::AccountPrivKey, transparent: legacy::AccountPrivKey,
sapling: sapling::ExtendedSpendingKey, sapling: sapling::ExtendedSpendingKey,
orchard: orchard::keys::SpendingKey,
} }
#[doc(hidden)] #[doc(hidden)]
@ -95,6 +97,10 @@ impl UnifiedSpendingKey {
panic!("ZIP 32 seeds MUST be at least 32 bytes"); panic!("ZIP 32 seeds MUST be at least 32 bytes");
} }
let orchard =
orchard::keys::SpendingKey::from_zip32_seed(seed, params.coin_type(), account.into())
.map_err(DerivationError::Orchard)?;
#[cfg(feature = "transparent-inputs")] #[cfg(feature = "transparent-inputs")]
let transparent = legacy::AccountPrivKey::from_seed(params, seed, account) let transparent = legacy::AccountPrivKey::from_seed(params, seed, account)
.map_err(DerivationError::Transparent)?; .map_err(DerivationError::Transparent)?;
@ -104,6 +110,7 @@ impl UnifiedSpendingKey {
#[cfg(feature = "transparent-inputs")] #[cfg(feature = "transparent-inputs")]
transparent, transparent,
sapling: sapling::spending_key(seed, params.coin_type(), account), sapling: sapling::spending_key(seed, params.coin_type(), account),
orchard,
}) })
} }
@ -112,7 +119,7 @@ impl UnifiedSpendingKey {
#[cfg(feature = "transparent-inputs")] #[cfg(feature = "transparent-inputs")]
transparent: Some(self.transparent.to_account_pubkey()), transparent: Some(self.transparent.to_account_pubkey()),
sapling: Some(sapling::ExtendedFullViewingKey::from(&self.sapling).into()), sapling: Some(sapling::ExtendedFullViewingKey::from(&self.sapling).into()),
orchard: None, orchard: Some((&self.orchard).into()),
unknown: vec![], unknown: vec![],
} }
} }
@ -128,11 +135,15 @@ impl UnifiedSpendingKey {
&self.transparent &self.transparent
} }
/// Returns the Sapling extended full viewing key component of this /// Returns the Sapling extended spending key component of this unified spending key.
/// unified key.
pub fn sapling(&self) -> &sapling::ExtendedSpendingKey { pub fn sapling(&self) -> &sapling::ExtendedSpendingKey {
&self.sapling &self.sapling
} }
/// Returns the Orchard spending key component of this unified spending key.
pub fn orchard(&self) -> &orchard::keys::SpendingKey {
&self.orchard
}
} }
/// A set of viewing keys that are all associated with a single /// A set of viewing keys that are all associated with a single
@ -283,6 +294,11 @@ impl UnifiedFullViewingKey {
self.sapling.as_ref() self.sapling.as_ref()
} }
/// Returns the Orchard full viewing key component of this unified key.
pub fn orchard(&self) -> Option<&orchard::keys::FullViewingKey> {
self.orchard.as_ref()
}
/// Attempts to derive the Unified Address for the given diversifier index. /// Attempts to derive the Unified Address for the given diversifier index.
/// ///
/// Returns `None` if the specified index does not produce a valid diversifier. /// Returns `None` if the specified index does not produce a valid diversifier.