use std::{
array::TryFromSliceError,
collections::{BTreeMap, BTreeSet, HashMap},
};
use orchard::tree::MerkleHashOrchard;
use task::ScanTask;
use tokio::sync::mpsc;
use incrementalmerkletree::Position;
use zcash_client_backend::proto::compact_formats::{CompactBlock, CompactTx};
use zcash_keys::keys::UnifiedFullViewingKey;
use zcash_primitives::{transaction::TxId, zip32::AccountId};
use zcash_protocol::consensus::{self, BlockHeight};
use crate::{
client::FetchRequest,
error::{ScanError, ServerError},
sync::ScanPriority,
wallet::{NullifierMap, OutputId, ScanTarget, WalletBlock, WalletTransaction},
witness::{self, LocatedTreeData, WitnessData},
};
use self::{compact_blocks::scan_compact_blocks, transactions::scan_transactions};
pub(crate) mod compact_blocks;
pub(crate) mod task;
pub(crate) mod transactions;
struct InitialScanData {
start_seam_block: Option<WalletBlock>,
end_seam_block: Option<WalletBlock>,
sapling_initial_tree_size: u32,
orchard_initial_tree_size: u32,
}
impl InitialScanData {
async fn new<P>(
fetch_request_sender: mpsc::UnboundedSender<FetchRequest>,
consensus_parameters: &P,
first_block: &CompactBlock,
start_seam_block: Option<WalletBlock>,
end_seam_block: Option<WalletBlock>,
) -> Result<Self, ServerError>
where
P: consensus::Parameters + Sync + Send + 'static,
{
let (sapling_initial_tree_size, orchard_initial_tree_size) =
if let Some(prev) = &start_seam_block {
(
prev.tree_bounds().sapling_final_tree_size,
prev.tree_bounds().orchard_final_tree_size,
)
} else {
let tree_bounds = compact_blocks::calculate_block_tree_bounds(
consensus_parameters,
fetch_request_sender,
first_block,
)
.await?;
(
tree_bounds.sapling_initial_tree_size,
tree_bounds.orchard_initial_tree_size,
)
};
Ok(InitialScanData {
start_seam_block,
end_seam_block,
sapling_initial_tree_size,
orchard_initial_tree_size,
})
}
}
struct ScanData {
nullifiers: NullifierMap,
wallet_blocks: BTreeMap<BlockHeight, WalletBlock>,
decrypted_scan_targets: BTreeSet<ScanTarget>,
decrypted_note_data: DecryptedNoteData,
witness_data: WitnessData,
}
pub(crate) struct ScanResults {
pub(crate) nullifiers: NullifierMap,
pub(crate) outpoints: BTreeMap<OutputId, ScanTarget>,
pub(crate) scanned_blocks: BTreeMap<BlockHeight, WalletBlock>,
pub(crate) wallet_transactions: HashMap<TxId, WalletTransaction>,
pub(crate) sapling_located_trees: Vec<LocatedTreeData<sapling_crypto::Node>>,
pub(crate) orchard_located_trees: Vec<LocatedTreeData<MerkleHashOrchard>>,
}
pub(crate) struct DecryptedNoteData {
sapling_nullifiers_and_positions: HashMap<OutputId, (sapling_crypto::Nullifier, Position)>,
orchard_nullifiers_and_positions: HashMap<OutputId, (orchard::note::Nullifier, Position)>,
}
impl DecryptedNoteData {
pub(crate) fn new() -> Self {
DecryptedNoteData {
sapling_nullifiers_and_positions: HashMap::new(),
orchard_nullifiers_and_positions: HashMap::new(),
}
}
}
pub(crate) async fn scan<P>(
fetch_request_sender: mpsc::UnboundedSender<FetchRequest>,
consensus_parameters: &P,
ufvks: &HashMap<AccountId, UnifiedFullViewingKey>,
scan_task: ScanTask,
max_batch_outputs: usize,
) -> Result<ScanResults, ScanError>
where
P: consensus::Parameters + Sync + Send + 'static,
{
let ScanTask {
compact_blocks,
scan_range,
start_seam_block,
end_seam_block,
mut scan_targets,
transparent_addresses,
} = scan_task;
if compact_blocks
.first()
.expect("compacts blocks should not be empty")
.height
!= u64::from(scan_range.block_range().start)
|| compact_blocks
.last()
.expect("compacts blocks should not be empty")
.height
!= u64::from(scan_range.block_range().end - 1)
{
panic!("compact blocks do not match scan range!")
}
if scan_range.priority() == ScanPriority::ScannedWithoutMapping {
let mut nullifiers = NullifierMap::new();
for block in &compact_blocks {
for transaction in &block.vtx {
collect_nullifiers(&mut nullifiers, block.height(), transaction)?;
}
}
return Ok(ScanResults {
nullifiers,
outpoints: BTreeMap::new(),
scanned_blocks: BTreeMap::new(),
wallet_transactions: HashMap::new(),
sapling_located_trees: Vec::new(),
orchard_located_trees: Vec::new(),
});
}
let initial_scan_data = InitialScanData::new(
fetch_request_sender.clone(),
consensus_parameters,
compact_blocks
.first()
.expect("compacts blocks should not be empty"),
start_seam_block,
end_seam_block,
)
.await?;
let consensus_parameters_clone = consensus_parameters.clone();
let ufvks_clone = ufvks.clone();
let scan_data = tokio::task::spawn_blocking(move || {
scan_compact_blocks(
compact_blocks,
&consensus_parameters_clone,
&ufvks_clone,
initial_scan_data,
max_batch_outputs / 8,
)
})
.await
.expect("task panicked")?;
let ScanData {
nullifiers,
wallet_blocks,
mut decrypted_scan_targets,
decrypted_note_data,
witness_data,
} = scan_data;
scan_targets.append(&mut decrypted_scan_targets);
let mut outpoints = BTreeMap::new();
let wallet_transactions = scan_transactions(
fetch_request_sender,
consensus_parameters,
ufvks,
scan_targets,
decrypted_note_data,
&wallet_blocks,
&mut outpoints,
transparent_addresses,
)
.await?;
let WitnessData {
sapling_initial_position,
orchard_initial_position,
sapling_leaves_and_retentions,
orchard_leaves_and_retentions,
} = witness_data;
let (sapling_located_trees, orchard_located_trees) = tokio::task::spawn_blocking(move || {
(
witness::build_located_trees(
sapling_initial_position,
sapling_leaves_and_retentions,
max_batch_outputs / 8,
),
witness::build_located_trees(
orchard_initial_position,
orchard_leaves_and_retentions,
max_batch_outputs / 8,
),
)
})
.await
.expect("task panicked");
Ok(ScanResults {
nullifiers,
outpoints,
scanned_blocks: wallet_blocks,
wallet_transactions,
sapling_located_trees,
orchard_located_trees,
})
}
fn collect_nullifiers(
nullifier_map: &mut NullifierMap,
block_height: BlockHeight,
transaction: &CompactTx,
) -> Result<(), ScanError> {
transaction
.spends
.iter()
.map(|spend| sapling_crypto::Nullifier::from_slice(spend.nf.as_slice()))
.collect::<Result<Vec<sapling_crypto::Nullifier>, TryFromSliceError>>()?
.into_iter()
.for_each(|nullifier| {
nullifier_map.sapling.insert(
nullifier,
ScanTarget {
block_height,
txid: transaction.txid(),
narrow_scan_area: false,
},
);
});
transaction
.actions
.iter()
.map(|action| {
orchard::note::Nullifier::from_bytes(
action.nullifier.as_slice().try_into().map_err(|_| {
ScanError::InvalidOrchardNullifierLength(action.nullifier.len())
})?,
)
.into_option()
.ok_or(ScanError::InvalidOrchardNullifier)
})
.collect::<Result<Vec<orchard::note::Nullifier>, ScanError>>()?
.into_iter()
.for_each(|nullifier| {
nullifier_map.orchard.insert(
nullifier,
ScanTarget {
block_height,
txid: transaction.txid(),
narrow_scan_area: false,
},
);
});
Ok(())
}