zcash_client_sqlite: Make `ephemeral_addresses` migration forward-compatible
This commit is contained in:
parent
e555854f87
commit
5bee983e6c
|
@ -81,7 +81,98 @@ impl<P: consensus::Parameters> RusqliteMigration for Migration<P> {
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod tests {
|
mod tests {
|
||||||
use crate::wallet::init::migrations::tests::test_migrate;
|
use rusqlite::{named_params, Connection};
|
||||||
|
use secrecy::{ExposeSecret, SecretVec};
|
||||||
|
use zcash_client_backend::data_api::AccountBirthday;
|
||||||
|
use zcash_keys::keys::UnifiedSpendingKey;
|
||||||
|
use zcash_protocol::consensus::Network;
|
||||||
|
use zip32::fingerprint::SeedFingerprint;
|
||||||
|
|
||||||
|
use crate::{
|
||||||
|
error::SqliteClientError,
|
||||||
|
wallet::{self, init::migrations::tests::test_migrate, transparent},
|
||||||
|
AccountId, WalletDb,
|
||||||
|
};
|
||||||
|
|
||||||
|
/// This is a minimized copy of [`wallet::create_account`] as of the time of the
|
||||||
|
/// creation of this migration.
|
||||||
|
fn create_account(
|
||||||
|
wdb: &mut WalletDb<Connection, Network>,
|
||||||
|
seed: &SecretVec<u8>,
|
||||||
|
birthday: &AccountBirthday,
|
||||||
|
) -> Result<(AccountId, UnifiedSpendingKey), SqliteClientError> {
|
||||||
|
wdb.transactionally(|wdb| {
|
||||||
|
let seed_fingerprint =
|
||||||
|
SeedFingerprint::from_seed(seed.expose_secret()).ok_or_else(|| {
|
||||||
|
SqliteClientError::BadAccountData(
|
||||||
|
"Seed must be between 32 and 252 bytes in length.".to_owned(),
|
||||||
|
)
|
||||||
|
})?;
|
||||||
|
let account_index = wallet::max_zip32_account_index(wdb.conn.0, &seed_fingerprint)?
|
||||||
|
.map(|a| a.next().ok_or(SqliteClientError::AccountIdOutOfRange))
|
||||||
|
.transpose()?
|
||||||
|
.unwrap_or(zip32::AccountId::ZERO);
|
||||||
|
|
||||||
|
let usk =
|
||||||
|
UnifiedSpendingKey::from_seed(&wdb.params, seed.expose_secret(), account_index)
|
||||||
|
.map_err(|_| SqliteClientError::KeyDerivationError(account_index))?;
|
||||||
|
let ufvk = usk.to_unified_full_viewing_key();
|
||||||
|
|
||||||
|
let orchard_item = ufvk.orchard().map(|k| k.to_bytes());
|
||||||
|
let sapling_item = ufvk.sapling().map(|k| k.to_bytes());
|
||||||
|
#[cfg(feature = "transparent-inputs")]
|
||||||
|
let transparent_item = ufvk.transparent().map(|k| k.serialize());
|
||||||
|
#[cfg(not(feature = "transparent-inputs"))]
|
||||||
|
let transparent_item: Option<Vec<u8>> = None;
|
||||||
|
|
||||||
|
let birthday_sapling_tree_size = Some(birthday.sapling_frontier().tree_size());
|
||||||
|
#[cfg(feature = "orchard")]
|
||||||
|
let birthday_orchard_tree_size = Some(birthday.orchard_frontier().tree_size());
|
||||||
|
#[cfg(not(feature = "orchard"))]
|
||||||
|
let birthday_orchard_tree_size: Option<u64> = None;
|
||||||
|
|
||||||
|
let account_id: AccountId = wdb.conn.0.query_row(
|
||||||
|
r#"
|
||||||
|
INSERT INTO accounts (
|
||||||
|
account_kind, hd_seed_fingerprint, hd_account_index,
|
||||||
|
ufvk, uivk,
|
||||||
|
orchard_fvk_item_cache, sapling_fvk_item_cache, p2pkh_fvk_item_cache,
|
||||||
|
birthday_height, birthday_sapling_tree_size, birthday_orchard_tree_size,
|
||||||
|
recover_until_height
|
||||||
|
)
|
||||||
|
VALUES (
|
||||||
|
:account_kind, :hd_seed_fingerprint, :hd_account_index,
|
||||||
|
:ufvk, :uivk,
|
||||||
|
:orchard_fvk_item_cache, :sapling_fvk_item_cache, :p2pkh_fvk_item_cache,
|
||||||
|
:birthday_height, :birthday_sapling_tree_size, :birthday_orchard_tree_size,
|
||||||
|
:recover_until_height
|
||||||
|
)
|
||||||
|
RETURNING id;
|
||||||
|
"#,
|
||||||
|
named_params![
|
||||||
|
":account_kind": 0, // 0 == Derived
|
||||||
|
":hd_seed_fingerprint": seed_fingerprint.to_bytes(),
|
||||||
|
":hd_account_index": u32::from(account_index),
|
||||||
|
":ufvk": ufvk.encode(&wdb.params),
|
||||||
|
":uivk": ufvk.to_unified_incoming_viewing_key().encode(&wdb.params),
|
||||||
|
":orchard_fvk_item_cache": orchard_item,
|
||||||
|
":sapling_fvk_item_cache": sapling_item,
|
||||||
|
":p2pkh_fvk_item_cache": transparent_item,
|
||||||
|
":birthday_height": u32::from(birthday.height()),
|
||||||
|
":birthday_sapling_tree_size": birthday_sapling_tree_size,
|
||||||
|
":birthday_orchard_tree_size": birthday_orchard_tree_size,
|
||||||
|
":recover_until_height": birthday.recover_until().map(u32::from)
|
||||||
|
],
|
||||||
|
|row| Ok(AccountId(row.get(0)?)),
|
||||||
|
)?;
|
||||||
|
|
||||||
|
// Initialize the `ephemeral_addresses` table.
|
||||||
|
#[cfg(feature = "transparent-inputs")]
|
||||||
|
transparent::ephemeral::init_account(&mut wdb.conn.0, &wdb.params, account_id)?;
|
||||||
|
|
||||||
|
Ok((account_id, usk))
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn migrate() {
|
fn migrate() {
|
||||||
|
@ -95,7 +186,7 @@ mod tests {
|
||||||
use secrecy::Secret;
|
use secrecy::Secret;
|
||||||
use tempfile::NamedTempFile;
|
use tempfile::NamedTempFile;
|
||||||
use zcash_client_backend::{
|
use zcash_client_backend::{
|
||||||
data_api::{AccountBirthday, AccountSource, WalletWrite},
|
data_api::{AccountBirthday, AccountSource},
|
||||||
wallet::TransparentAddressMetadata,
|
wallet::TransparentAddressMetadata,
|
||||||
};
|
};
|
||||||
use zcash_keys::keys::UnifiedSpendingKey;
|
use zcash_keys::keys::UnifiedSpendingKey;
|
||||||
|
@ -192,9 +283,8 @@ mod tests {
|
||||||
|
|
||||||
// Creating a new account should initialize `ephemeral_addresses` for that account.
|
// Creating a new account should initialize `ephemeral_addresses` for that account.
|
||||||
let seed1 = vec![0x01; 32];
|
let seed1 = vec![0x01; 32];
|
||||||
let (account1_id, _usk) = db_data
|
let (account1_id, _usk) =
|
||||||
.create_account(&Secret::new(seed1), &birthday)
|
create_account(&mut db_data, &Secret::new(seed1), &birthday).unwrap();
|
||||||
.unwrap();
|
|
||||||
assert_ne!(account0_id, account1_id);
|
assert_ne!(account0_id, account1_id);
|
||||||
check(&db_data, account1_id);
|
check(&db_data, account1_id);
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue