Skip to content

Commit

Permalink
zcash_keys: Add ReceiverRequirement enum.
Browse files Browse the repository at this point in the history
This permits `UnifiedAddressRequest` values an additional dimension of
flexibility, permitting generation of unified addresses having receivers
for all recever types for which a key item exists and a diversifier
index is valid.
  • Loading branch information
nuttycom committed Dec 26, 2024
1 parent fe5c33e commit 9cc97e0
Show file tree
Hide file tree
Showing 8 changed files with 201 additions and 114 deletions.
11 changes: 7 additions & 4 deletions zcash_client_sqlite/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -93,6 +93,9 @@ use {
zcash_keys::encoding::AddressCodec,
};

#[cfg(any(feature = "orchard", feature = "transparent-inputs"))]
use zcash_keys::keys::ReceiverRequirement;

#[cfg(feature = "multicore")]
use maybe_rayon::{
prelude::{IndexedParallelIterator, ParallelIterator},
Expand Down Expand Up @@ -150,14 +153,14 @@ pub(crate) const SAPLING_TABLES_PREFIX: &str = "sapling";
pub(crate) const ORCHARD_TABLES_PREFIX: &str = "orchard";

#[cfg(not(feature = "orchard"))]
pub(crate) const UA_ORCHARD: bool = false;
pub(crate) const UA_ORCHARD: ReceiverRequirement = ReceiverRequirement::Omit;
#[cfg(feature = "orchard")]
pub(crate) const UA_ORCHARD: bool = true;
pub(crate) const UA_ORCHARD: ReceiverRequirement = ReceiverRequirement::Require;

#[cfg(not(feature = "transparent-inputs"))]
pub(crate) const UA_TRANSPARENT: bool = false;
pub(crate) const UA_TRANSPARENT: ReceiverRequirement = ReceiverRequirement::Omit;
#[cfg(feature = "transparent-inputs")]
pub(crate) const UA_TRANSPARENT: bool = true;
pub(crate) const UA_TRANSPARENT: ReceiverRequirement = ReceiverRequirement::Require;

/// Unique identifier for a specific account tracked by a [`WalletDb`].
///
Expand Down
5 changes: 1 addition & 4 deletions zcash_client_sqlite/src/wallet.rs
Original file line number Diff line number Diff line change
Expand Up @@ -294,10 +294,7 @@ pub(crate) fn seed_matches_derived_account<P: consensus::Parameters>(
let usk = UnifiedSpendingKey::from_seed(params, &seed.expose_secret()[..], account_index)
.map_err(|_| SqliteClientError::KeyDerivationError(account_index))?;

let (seed_addr, _) = usk.to_unified_full_viewing_key().default_address(Some(
UnifiedAddressRequest::all().expect("At least one supported pool feature is enabled."),
))?;

let (seed_addr, _) = usk.to_unified_full_viewing_key().default_address(None)?;
let (uivk_addr, _) = uivk.default_address(None)?;

#[cfg(not(feature = "orchard"))]
Expand Down
9 changes: 6 additions & 3 deletions zcash_client_sqlite/src/wallet/init.rs
Original file line number Diff line number Diff line change
Expand Up @@ -444,7 +444,10 @@ mod tests {
address::Address,
data_api::testing::TestBuilder,
encoding::{encode_extended_full_viewing_key, encode_payment_address},
keys::{sapling, UnifiedAddressRequest, UnifiedFullViewingKey, UnifiedSpendingKey},
keys::{
sapling, ReceiverRequirement::*, UnifiedAddressRequest, UnifiedFullViewingKey,
UnifiedSpendingKey,
},
};

use ::sapling::zip32::ExtendedFullViewingKey;
Expand Down Expand Up @@ -989,7 +992,7 @@ mod tests {

// Unified addresses at the time of the addition of migrations did not contain an
// Orchard component.
let ua_request = UnifiedAddressRequest::unsafe_new(false, true, UA_TRANSPARENT);
let ua_request = UnifiedAddressRequest::unsafe_new(Omit, Require, UA_TRANSPARENT);
let address_str = Address::Unified(
ufvk.default_address(Some(ua_request))
.expect("A valid default address exists for the UFVK")
Expand Down Expand Up @@ -1116,7 +1119,7 @@ mod tests {
assert_eq!(tv.unified_addr, ua.encode(&Network::MainNetwork));

// hardcoded with knowledge of what's coming next
let ua_request = UnifiedAddressRequest::unsafe_new(false, true, true);
let ua_request = UnifiedAddressRequest::unsafe_new(Omit, Require, Require);
db_data
.get_next_available_address(account_id, Some(ua_request))
.unwrap()
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -394,10 +394,10 @@ mod tests {
#[test]
#[cfg(feature = "transparent-inputs")]
fn migrate_from_wm2() {
use ::transparent::keys::NonHardenedChildIndex;
use zcash_client_backend::keys::UnifiedAddressRequest;
use zcash_primitives::{
legacy::keys::NonHardenedChildIndex, transaction::components::amount::NonNegativeAmount,
};
use zcash_keys::keys::ReceiverRequirement::*;
use zcash_protocol::value::Zatoshis as NonNegativeAmount;

use crate::UA_TRANSPARENT;

Expand Down Expand Up @@ -444,8 +444,8 @@ mod tests {
let ufvk = usk.to_unified_full_viewing_key();
let (ua, _) = ufvk
.default_address(Some(UnifiedAddressRequest::unsafe_new(
false,
true,
Omit,
Require,
UA_TRANSPARENT,
)))
.expect("A valid default address exists for the UFVK");
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,11 @@ use rusqlite::{named_params, Transaction};
use schemerz_rusqlite::RusqliteMigration;
use uuid::Uuid;
use zcash_client_backend::{address::Address, keys::UnifiedFullViewingKey};
use zcash_keys::{address::UnifiedAddress, encoding::AddressCodec, keys::UnifiedAddressRequest};
use zcash_keys::{
address::UnifiedAddress,
encoding::AddressCodec,
keys::{ReceiverRequirement::*, UnifiedAddressRequest},
};
use zcash_primitives::consensus;
use zip32::{AccountId, DiversifierIndex};

Expand Down Expand Up @@ -87,7 +91,7 @@ impl<P: consensus::Parameters> RusqliteMigration for Migration<P> {
));
};
let (expected_address, idx) = ufvk.default_address(Some(
UnifiedAddressRequest::unsafe_new(false, true, UA_TRANSPARENT),
UnifiedAddressRequest::unsafe_new(Omit, Require, UA_TRANSPARENT),
))?;
if decoded_address != expected_address {
return Err(WalletMigrationError::CorruptedData(format!(
Expand Down Expand Up @@ -159,7 +163,7 @@ impl<P: consensus::Parameters> RusqliteMigration for Migration<P> {
)?;

let (address, d_idx) = ufvk.default_address(Some(
UnifiedAddressRequest::unsafe_new(false, true, UA_TRANSPARENT),
UnifiedAddressRequest::unsafe_new(Omit, Require, UA_TRANSPARENT),
))?;
insert_address(transaction, &self.params, account, d_idx, &address)?;
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@ use schemerz_rusqlite::RusqliteMigration;
use uuid::Uuid;

use zcash_client_backend::keys::{
UnifiedAddressRequest, UnifiedFullViewingKey, UnifiedIncomingViewingKey,
ReceiverRequirement::*, UnifiedAddressRequest, UnifiedFullViewingKey, UnifiedIncomingViewingKey,
};
use zcash_primitives::consensus;

Expand Down Expand Up @@ -66,7 +66,7 @@ impl<P: consensus::Parameters> RusqliteMigration for Migration<P> {
};

let (default_addr, diversifier_index) = uivk.default_address(Some(
UnifiedAddressRequest::unsafe_new(UA_ORCHARD, true, UA_TRANSPARENT),
UnifiedAddressRequest::unsafe_new(UA_ORCHARD, Require, UA_TRANSPARENT),
))?;

let mut di_be = *diversifier_index.as_bytes();
Expand All @@ -92,7 +92,9 @@ mod tests {
use secrecy::SecretVec;
use tempfile::NamedTempFile;

use zcash_client_backend::keys::{UnifiedAddressRequest, UnifiedSpendingKey};
use zcash_client_backend::keys::{
ReceiverRequirement::*, UnifiedAddressRequest, UnifiedSpendingKey,
};
use zcash_keys::address::Address;
use zcash_primitives::consensus::Network;

Expand Down Expand Up @@ -141,8 +143,8 @@ mod tests {

let (addr, diversifier_index) = ufvk
.default_address(Some(UnifiedAddressRequest::unsafe_new(
false,
true,
Omit,
Require,
UA_TRANSPARENT,
)))
.unwrap();
Expand Down Expand Up @@ -170,7 +172,7 @@ mod tests {
Ok(Address::Unified(ua)) => {
assert!(!ua.has_orchard());
assert!(ua.has_sapling());
assert_eq!(ua.has_transparent(), UA_TRANSPARENT);
assert_eq!(ua.has_transparent(), UA_TRANSPARENT == Require);
}
other => panic!("Unexpected result from address decoding: {:?}", other),
}
Expand All @@ -186,9 +188,9 @@ mod tests {
Ok(Address::decode(&db_data.params, &row.get::<_, String>(0)?).unwrap())
}) {
Ok(Address::Unified(ua)) => {
assert_eq!(ua.has_orchard(), UA_ORCHARD);
assert_eq!(ua.has_orchard(), UA_ORCHARD == Require);
assert!(ua.has_sapling());
assert_eq!(ua.has_transparent(), UA_TRANSPARENT);
assert_eq!(ua.has_transparent(), UA_TRANSPARENT == Require);
}
other => panic!("Unexpected result from address decoding: {:?}", other),
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@ use secrecy::{ExposeSecret, SecretVec};
use uuid::Uuid;

use zcash_client_backend::{address::Address, keys::UnifiedSpendingKey, PoolType};
use zcash_keys::keys::UnifiedAddressRequest;
use zcash_keys::keys::{ReceiverRequirement::*, UnifiedAddressRequest};
use zcash_primitives::{consensus, zip32::AccountId};

#[cfg(feature = "transparent-inputs")]
Expand Down Expand Up @@ -82,7 +82,7 @@ impl<P: consensus::Parameters> RusqliteMigration for Migration<P> {
// our second assumption above, and we report this as corrupted data.
let mut seed_is_relevant = false;

let ua_request = UnifiedAddressRequest::unsafe_new(false, true, UA_TRANSPARENT);
let ua_request = UnifiedAddressRequest::unsafe_new(Omit, Require, UA_TRANSPARENT);
let mut rows = stmt_fetch_accounts.query([])?;
while let Some(row) = rows.next()? {
// We only need to check for the presence of the seed if we have keys that
Expand Down
Loading

0 comments on commit 9cc97e0

Please sign in to comment.