use crate::{
atomic_batch_scope,
cow_to_cloned,
cow_to_copied,
helpers::{Map, MapRead},
TransactionStorage,
TransactionStore,
TransitionStorage,
TransitionStore,
};
use console::{
network::prelude::*,
program::{BlockTree, HeaderLeaf, ProgramID, StatePath},
types::Field,
};
use ledger_authority::Authority;
use ledger_block::{
Block,
ConfirmedTransaction,
Header,
NumFinalizeSize,
Ratifications,
Rejected,
Transaction,
Transactions,
};
use ledger_coinbase::{CoinbaseSolution, ProverSolution, PuzzleCommitment};
use ledger_narwhal_batch_certificate::BatchCertificate;
use synthesizer_program::Program;
use anyhow::Result;
use parking_lot::RwLock;
use std::{borrow::Cow, io::Cursor, sync::Arc};
#[cfg(not(feature = "serial"))]
use rayon::prelude::*;
#[derive(Copy, Clone, PartialEq, Eq, Hash, Serialize, Deserialize)]
pub enum ConfirmedTxType {
AcceptedDeploy(u32),
AcceptedExecute(u32),
RejectedDeploy(u32),
RejectedExecute(u32),
}
#[allow(clippy::type_complexity)]
fn to_confirmed_tuple<N: Network>(
confirmed: ConfirmedTransaction<N>,
) -> Result<(ConfirmedTxType, Transaction<N>, Vec<u8>, Option<Rejected<N>>)> {
match confirmed {
ConfirmedTransaction::AcceptedDeploy(index, tx, finalize) => {
let num_finalize = NumFinalizeSize::try_from(finalize.len())?;
Ok((ConfirmedTxType::AcceptedDeploy(index), tx, (num_finalize, finalize).to_bytes_le()?, None))
}
ConfirmedTransaction::AcceptedExecute(index, tx, finalize) => {
let num_finalize = NumFinalizeSize::try_from(finalize.len())?;
Ok((ConfirmedTxType::AcceptedExecute(index), tx, (num_finalize, finalize).to_bytes_le()?, None))
}
ConfirmedTransaction::RejectedDeploy(index, tx, rejected, finalize) => {
let num_finalize = NumFinalizeSize::try_from(finalize.len())?;
let mut blob = Vec::new();
rejected.write_le(&mut blob)?;
num_finalize.write_le(&mut blob)?;
finalize.write_le(&mut blob)?;
Ok((ConfirmedTxType::RejectedDeploy(index), tx, blob, Some(rejected)))
}
ConfirmedTransaction::RejectedExecute(index, tx, rejected, finalize) => {
let num_finalize = NumFinalizeSize::try_from(finalize.len())?;
let mut blob = Vec::new();
rejected.write_le(&mut blob)?;
num_finalize.write_le(&mut blob)?;
finalize.write_le(&mut blob)?;
Ok((ConfirmedTxType::RejectedExecute(index), tx, blob, Some(rejected)))
}
}
}
fn to_confirmed_transaction<N: Network>(
confirmed_type: ConfirmedTxType,
transaction: Transaction<N>,
blob: Vec<u8>,
) -> Result<ConfirmedTransaction<N>> {
match confirmed_type {
ConfirmedTxType::AcceptedDeploy(index) => {
let mut cursor = Cursor::new(blob);
let num_finalize = NumFinalizeSize::read_le(&mut cursor)?;
let finalize = (0..num_finalize).map(|_| FromBytes::read_le(&mut cursor)).collect::<Result<Vec<_>, _>>()?;
ConfirmedTransaction::accepted_deploy(index, transaction, finalize)
}
ConfirmedTxType::AcceptedExecute(index) => {
let mut cursor = Cursor::new(blob);
let num_finalize = NumFinalizeSize::read_le(&mut cursor)?;
let finalize = (0..num_finalize).map(|_| FromBytes::read_le(&mut cursor)).collect::<Result<Vec<_>, _>>()?;
ConfirmedTransaction::accepted_execute(index, transaction, finalize)
}
ConfirmedTxType::RejectedDeploy(index) => {
let mut cursor = Cursor::new(blob);
let rejected = Rejected::read_le(&mut cursor)?;
let num_finalize = NumFinalizeSize::read_le(&mut cursor)?;
let finalize = (0..num_finalize).map(|_| FromBytes::read_le(&mut cursor)).collect::<Result<Vec<_>, _>>()?;
ConfirmedTransaction::rejected_deploy(index, transaction, rejected, finalize)
}
ConfirmedTxType::RejectedExecute(index) => {
let mut cursor = Cursor::new(blob);
let rejected = Rejected::read_le(&mut cursor)?;
let num_finalize = NumFinalizeSize::read_le(&mut cursor)?;
let finalize = (0..num_finalize).map(|_| FromBytes::read_le(&mut cursor)).collect::<Result<Vec<_>, _>>()?;
ConfirmedTransaction::rejected_execute(index, transaction, rejected, finalize)
}
}
}
pub trait BlockStorage<N: Network>: 'static + Clone + Send + Sync {
type StateRootMap: for<'a> Map<'a, u32, N::StateRoot>;
type ReverseStateRootMap: for<'a> Map<'a, N::StateRoot, u32>;
type IDMap: for<'a> Map<'a, u32, N::BlockHash>;
type ReverseIDMap: for<'a> Map<'a, N::BlockHash, u32>;
type HeaderMap: for<'a> Map<'a, N::BlockHash, Header<N>>;
type AuthorityMap: for<'a> Map<'a, N::BlockHash, Authority<N>>;
type CertificateMap: for<'a> Map<'a, Field<N>, (u32, u64)>;
type RatificationsMap: for<'a> Map<'a, N::BlockHash, Ratifications<N>>;
type SolutionsMap: for<'a> Map<'a, N::BlockHash, Option<CoinbaseSolution<N>>>;
type PuzzleCommitmentsMap: for<'a> Map<'a, PuzzleCommitment<N>, u32>;
type TransactionsMap: for<'a> Map<'a, N::BlockHash, Vec<N::TransactionID>>;
type AbortedTransactionIDsMap: for<'a> Map<'a, N::BlockHash, Vec<N::TransactionID>>;
type RejectedOrAbortedTransactionIDMap: for<'a> Map<'a, N::TransactionID, N::BlockHash>;
type ConfirmedTransactionsMap: for<'a> Map<'a, N::TransactionID, (N::BlockHash, ConfirmedTxType, Vec<u8>)>;
type RejectedDeploymentOrExecutionMap: for<'a> Map<'a, Field<N>, Rejected<N>>;
type TransactionStorage: TransactionStorage<N, TransitionStorage = Self::TransitionStorage>;
type TransitionStorage: TransitionStorage<N>;
fn open(dev: Option<u16>) -> Result<Self>;
fn state_root_map(&self) -> &Self::StateRootMap;
fn reverse_state_root_map(&self) -> &Self::ReverseStateRootMap;
fn id_map(&self) -> &Self::IDMap;
fn reverse_id_map(&self) -> &Self::ReverseIDMap;
fn header_map(&self) -> &Self::HeaderMap;
fn authority_map(&self) -> &Self::AuthorityMap;
fn certificate_map(&self) -> &Self::CertificateMap;
fn ratifications_map(&self) -> &Self::RatificationsMap;
fn solutions_map(&self) -> &Self::SolutionsMap;
fn puzzle_commitments_map(&self) -> &Self::PuzzleCommitmentsMap;
fn transactions_map(&self) -> &Self::TransactionsMap;
fn aborted_transaction_ids_map(&self) -> &Self::AbortedTransactionIDsMap;
fn rejected_or_aborted_transaction_id_map(&self) -> &Self::RejectedOrAbortedTransactionIDMap;
fn confirmed_transactions_map(&self) -> &Self::ConfirmedTransactionsMap;
fn rejected_deployment_or_execution_map(&self) -> &Self::RejectedDeploymentOrExecutionMap;
fn transaction_store(&self) -> &TransactionStore<N, Self::TransactionStorage>;
fn transition_store(&self) -> &TransitionStore<N, Self::TransitionStorage> {
self.transaction_store().transition_store()
}
fn dev(&self) -> Option<u16> {
debug_assert!(self.transaction_store().dev() == self.transition_store().dev());
self.transition_store().dev()
}
fn start_atomic(&self) {
self.state_root_map().start_atomic();
self.reverse_state_root_map().start_atomic();
self.id_map().start_atomic();
self.reverse_id_map().start_atomic();
self.header_map().start_atomic();
self.authority_map().start_atomic();
self.certificate_map().start_atomic();
self.ratifications_map().start_atomic();
self.solutions_map().start_atomic();
self.puzzle_commitments_map().start_atomic();
self.transactions_map().start_atomic();
self.aborted_transaction_ids_map().start_atomic();
self.rejected_or_aborted_transaction_id_map().start_atomic();
self.confirmed_transactions_map().start_atomic();
self.rejected_deployment_or_execution_map().start_atomic();
self.transaction_store().start_atomic();
}
fn is_atomic_in_progress(&self) -> bool {
self.state_root_map().is_atomic_in_progress()
|| self.reverse_state_root_map().is_atomic_in_progress()
|| self.id_map().is_atomic_in_progress()
|| self.reverse_id_map().is_atomic_in_progress()
|| self.header_map().is_atomic_in_progress()
|| self.authority_map().is_atomic_in_progress()
|| self.certificate_map().is_atomic_in_progress()
|| self.ratifications_map().is_atomic_in_progress()
|| self.solutions_map().is_atomic_in_progress()
|| self.puzzle_commitments_map().is_atomic_in_progress()
|| self.transactions_map().is_atomic_in_progress()
|| self.aborted_transaction_ids_map().is_atomic_in_progress()
|| self.rejected_or_aborted_transaction_id_map().is_atomic_in_progress()
|| self.confirmed_transactions_map().is_atomic_in_progress()
|| self.rejected_deployment_or_execution_map().is_atomic_in_progress()
|| self.transaction_store().is_atomic_in_progress()
}
fn atomic_checkpoint(&self) {
self.state_root_map().atomic_checkpoint();
self.reverse_state_root_map().atomic_checkpoint();
self.id_map().atomic_checkpoint();
self.reverse_id_map().atomic_checkpoint();
self.header_map().atomic_checkpoint();
self.authority_map().atomic_checkpoint();
self.certificate_map().atomic_checkpoint();
self.ratifications_map().atomic_checkpoint();
self.solutions_map().atomic_checkpoint();
self.puzzle_commitments_map().atomic_checkpoint();
self.transactions_map().atomic_checkpoint();
self.aborted_transaction_ids_map().atomic_checkpoint();
self.rejected_or_aborted_transaction_id_map().atomic_checkpoint();
self.confirmed_transactions_map().atomic_checkpoint();
self.rejected_deployment_or_execution_map().atomic_checkpoint();
self.transaction_store().atomic_checkpoint();
}
fn clear_latest_checkpoint(&self) {
self.state_root_map().clear_latest_checkpoint();
self.reverse_state_root_map().clear_latest_checkpoint();
self.id_map().clear_latest_checkpoint();
self.reverse_id_map().clear_latest_checkpoint();
self.header_map().clear_latest_checkpoint();
self.authority_map().clear_latest_checkpoint();
self.certificate_map().clear_latest_checkpoint();
self.ratifications_map().clear_latest_checkpoint();
self.solutions_map().clear_latest_checkpoint();
self.puzzle_commitments_map().clear_latest_checkpoint();
self.transactions_map().clear_latest_checkpoint();
self.aborted_transaction_ids_map().clear_latest_checkpoint();
self.rejected_or_aborted_transaction_id_map().clear_latest_checkpoint();
self.confirmed_transactions_map().clear_latest_checkpoint();
self.rejected_deployment_or_execution_map().clear_latest_checkpoint();
self.transaction_store().clear_latest_checkpoint();
}
fn atomic_rewind(&self) {
self.state_root_map().atomic_rewind();
self.reverse_state_root_map().atomic_rewind();
self.id_map().atomic_rewind();
self.reverse_id_map().atomic_rewind();
self.header_map().atomic_rewind();
self.authority_map().atomic_rewind();
self.certificate_map().atomic_rewind();
self.ratifications_map().atomic_rewind();
self.solutions_map().atomic_rewind();
self.puzzle_commitments_map().atomic_rewind();
self.transactions_map().atomic_rewind();
self.aborted_transaction_ids_map().atomic_rewind();
self.rejected_or_aborted_transaction_id_map().atomic_rewind();
self.confirmed_transactions_map().atomic_rewind();
self.rejected_deployment_or_execution_map().atomic_rewind();
self.transaction_store().atomic_rewind();
}
fn abort_atomic(&self) {
self.state_root_map().abort_atomic();
self.reverse_state_root_map().abort_atomic();
self.id_map().abort_atomic();
self.reverse_id_map().abort_atomic();
self.header_map().abort_atomic();
self.authority_map().abort_atomic();
self.certificate_map().abort_atomic();
self.ratifications_map().abort_atomic();
self.solutions_map().abort_atomic();
self.puzzle_commitments_map().abort_atomic();
self.transactions_map().abort_atomic();
self.aborted_transaction_ids_map().abort_atomic();
self.rejected_or_aborted_transaction_id_map().abort_atomic();
self.confirmed_transactions_map().abort_atomic();
self.rejected_deployment_or_execution_map().abort_atomic();
self.transaction_store().abort_atomic();
}
fn finish_atomic(&self) -> Result<()> {
self.state_root_map().finish_atomic()?;
self.reverse_state_root_map().finish_atomic()?;
self.id_map().finish_atomic()?;
self.reverse_id_map().finish_atomic()?;
self.header_map().finish_atomic()?;
self.authority_map().finish_atomic()?;
self.certificate_map().finish_atomic()?;
self.ratifications_map().finish_atomic()?;
self.solutions_map().finish_atomic()?;
self.puzzle_commitments_map().finish_atomic()?;
self.transactions_map().finish_atomic()?;
self.aborted_transaction_ids_map().finish_atomic()?;
self.rejected_or_aborted_transaction_id_map().finish_atomic()?;
self.confirmed_transactions_map().finish_atomic()?;
self.rejected_deployment_or_execution_map().finish_atomic()?;
self.transaction_store().finish_atomic()
}
fn insert(&self, state_root: N::StateRoot, block: &Block<N>) -> Result<()> {
let confirmed = block
.transactions()
.iter()
.cloned()
.map(|confirmed| to_confirmed_tuple(confirmed))
.collect::<Result<Vec<_>, _>>()?;
let certificates_to_store = match block.authority() {
Authority::Beacon(_) => Vec::new(),
Authority::Quorum(subdag) => {
subdag.iter().flat_map(|(round, certificates)| certificates.iter().map(|c| (c.id(), *round))).collect()
}
};
let rejected_transaction_ids: Vec<_> = block
.transactions()
.iter()
.filter(|tx| tx.is_rejected())
.map(|tx| tx.to_unconfirmed_transaction_id())
.collect::<Result<Vec<_>>>()?;
atomic_batch_scope!(self, {
self.state_root_map().insert(block.height(), state_root)?;
self.reverse_state_root_map().insert(state_root, block.height())?;
self.id_map().insert(block.height(), block.hash())?;
self.reverse_id_map().insert(block.hash(), block.height())?;
self.header_map().insert(block.hash(), *block.header())?;
self.authority_map().insert(block.hash(), block.authority().clone())?;
for (certificate_id, round) in certificates_to_store {
self.certificate_map().insert(certificate_id, (block.height(), round))?;
}
self.ratifications_map().insert(block.hash(), block.ratifications().clone())?;
self.solutions_map().insert(block.hash(), block.solutions().cloned())?;
if let Some(solutions) = block.solutions() {
for puzzle_commitment in solutions.keys() {
self.puzzle_commitments_map().insert(*puzzle_commitment, block.height())?;
}
}
self.transactions_map().insert(block.hash(), block.transaction_ids().copied().collect())?;
self.aborted_transaction_ids_map().insert(block.hash(), block.aborted_transaction_ids().clone())?;
for aborted_transaction_id in block.aborted_transaction_ids() {
self.rejected_or_aborted_transaction_id_map().insert(*aborted_transaction_id, block.hash())?;
}
for rejected_transaction_id in rejected_transaction_ids {
self.rejected_or_aborted_transaction_id_map().insert(rejected_transaction_id, block.hash())?;
}
for (confirmed_type, transaction, blob, rejected) in confirmed {
self.confirmed_transactions_map().insert(transaction.id(), (block.hash(), confirmed_type, blob))?;
if let Some(rejected) = rejected {
self.rejected_deployment_or_execution_map().insert(rejected.to_id()?, rejected)?;
}
self.transaction_store().insert(&transaction)?;
}
Ok(())
})
}
fn remove(&self, block_hash: &N::BlockHash) -> Result<()> {
let block_height = match self.get_block_height(block_hash)? {
Some(height) => height,
None => bail!("Failed to remove block: missing block height for block hash '{block_hash}'"),
};
let state_root = match self.state_root_map().get_confirmed(&block_height)? {
Some(state_root) => cow_to_copied!(state_root),
None => bail!("Failed to remove block: missing state root for block height '{block_height}'"),
};
let transaction_ids = match self.transactions_map().get_confirmed(block_hash)? {
Some(transaction_ids) => transaction_ids,
None => bail!("Failed to remove block: missing transactions for block '{block_height}' ('{block_hash}')"),
};
let solutions = match self.solutions_map().get_confirmed(block_hash)? {
Some(solutions) => cow_to_cloned!(solutions),
None => {
bail!("Failed to remove block: missing solutions for block '{block_height}' ('{block_hash}')")
}
};
let aborted_transaction_ids = match self.get_block_aborted_transaction_ids(block_hash)? {
Some(transaction_ids) => transaction_ids,
None => Vec::new(),
};
let rejected_transaction_ids_and_deployment_or_execution_id = match self.get_block_transactions(block_hash)? {
Some(transactions) => transactions
.iter()
.filter(|tx| tx.is_rejected())
.map(|tx| Ok((tx.to_unconfirmed_transaction_id()?, tx.to_rejected_id()?)))
.collect::<Result<Vec<_>>>()?,
None => Vec::new(),
};
let certificate_ids_to_remove = match self.authority_map().get_confirmed(block_hash)? {
Some(authority) => match authority {
Cow::Owned(Authority::Beacon(_)) | Cow::Borrowed(Authority::Beacon(_)) => Vec::new(),
Cow::Owned(Authority::Quorum(ref subdag)) | Cow::Borrowed(Authority::Quorum(ref subdag)) => {
subdag.values().flatten().map(|c| c.id()).collect()
}
},
None => bail!("Failed to remove block: missing authority for block '{block_height}' ('{block_hash}')"),
};
atomic_batch_scope!(self, {
self.state_root_map().remove(&block_height)?;
self.reverse_state_root_map().remove(&state_root)?;
self.id_map().remove(&block_height)?;
self.reverse_id_map().remove(block_hash)?;
self.header_map().remove(block_hash)?;
self.authority_map().remove(block_hash)?;
for certificate_id in certificate_ids_to_remove.iter() {
self.certificate_map().remove(certificate_id)?;
}
self.ratifications_map().remove(block_hash)?;
self.solutions_map().remove(block_hash)?;
if let Some(solutions) = solutions {
for puzzle_commitment in solutions.keys() {
self.puzzle_commitments_map().remove(puzzle_commitment)?;
}
}
self.transactions_map().remove(block_hash)?;
self.aborted_transaction_ids_map().remove(block_hash)?;
for aborted_transaction_id in aborted_transaction_ids {
self.rejected_or_aborted_transaction_id_map().remove(&aborted_transaction_id)?;
}
for (rejected_transaction_id, rejected_id) in rejected_transaction_ids_and_deployment_or_execution_id {
self.rejected_or_aborted_transaction_id_map().remove(&rejected_transaction_id)?;
if let Some(rejected_id) = rejected_id {
self.rejected_deployment_or_execution_map().remove(&rejected_id)?;
}
}
for transaction_id in transaction_ids.iter() {
self.confirmed_transactions_map().remove(transaction_id)?;
self.transaction_store().remove(transaction_id)?;
}
Ok(())
})
}
fn contains_transaction_id(&self, transaction_id: &N::TransactionID) -> Result<bool> {
Ok(self.transaction_store().contains_transaction_id(transaction_id)?
|| self.contains_rejected_or_aborted_transaction_id(transaction_id)?)
}
fn contains_rejected_or_aborted_transaction_id(&self, transaction_id: &N::TransactionID) -> Result<bool> {
self.rejected_or_aborted_transaction_id_map().contains_key_confirmed(transaction_id)
}
fn contains_rejected_deployment_or_execution_id(&self, rejected_id: &Field<N>) -> Result<bool> {
self.rejected_deployment_or_execution_map().contains_key_confirmed(rejected_id)
}
fn find_block_height_from_state_root(&self, state_root: N::StateRoot) -> Result<Option<u32>> {
match self.reverse_state_root_map().get_confirmed(&state_root)? {
Some(block_height) => Ok(Some(cow_to_copied!(block_height))),
None => Ok(None),
}
}
fn find_block_hash(&self, transaction_id: &N::TransactionID) -> Result<Option<N::BlockHash>> {
match self.confirmed_transactions_map().get_confirmed(transaction_id)? {
Some(Cow::Borrowed((block_hash, _, _))) => Ok(Some(*block_hash)),
Some(Cow::Owned((block_hash, _, _))) => Ok(Some(block_hash)),
None => Ok(None),
}
}
fn find_block_height_from_puzzle_commitment(&self, puzzle_commitment: &PuzzleCommitment<N>) -> Result<Option<u32>> {
match self.puzzle_commitments_map().get_confirmed(puzzle_commitment)? {
Some(block_height) => Ok(Some(cow_to_copied!(block_height))),
None => Ok(None),
}
}
fn get_state_root(&self, block_height: u32) -> Result<Option<N::StateRoot>> {
match self.state_root_map().get_confirmed(&block_height)? {
Some(state_root) => Ok(Some(cow_to_copied!(state_root))),
None => Ok(None),
}
}
fn get_state_path_for_commitment(&self, commitment: &Field<N>, block_tree: &BlockTree<N>) -> Result<StatePath<N>> {
if !self.transition_store().contains_commitment(commitment)? {
bail!("Commitment '{commitment}' does not exist");
}
let transition_id = self.transition_store().find_transition_id(commitment)?;
let transaction_id = match self.transaction_store().find_transaction_id_from_transition_id(&transition_id)? {
Some(transaction_id) => transaction_id,
None => bail!("The transaction ID for commitment '{commitment}' is missing in storage"),
};
let block_hash = match self.find_block_hash(&transaction_id)? {
Some(block_hash) => block_hash,
None => bail!("The block hash for commitment '{commitment}' is missing in storage"),
};
let transition = match self.transition_store().get_transition(&transition_id)? {
Some(transition) => transition,
None => bail!("The transition '{transition_id}' for commitment '{commitment}' is missing in storage"),
};
let block = match self.get_block(&block_hash)? {
Some(block) => block,
None => bail!("The block '{block_hash}' for commitment '{commitment}' is missing in storage"),
};
let global_state_root = *block_tree.root();
let block_path = block_tree.prove(block.height() as usize, &block.hash().to_bits_le())?;
if !self.reverse_state_root_map().contains_key_confirmed(&global_state_root.into())? {
bail!("The global state root '{global_state_root}' for commitment '{commitment}' is missing in storage");
}
let transition_root = transition.to_root()?;
let transition_leaf = transition.to_leaf(commitment, false)?;
let transition_path = transition.to_path(&transition_leaf)?;
let transactions = block.transactions();
let transactions_path = match transactions.to_path(transaction_id) {
Ok(transactions_path) => transactions_path,
Err(_) => bail!("The transaction '{transaction_id}' for commitment '{commitment}' is not in the block"),
};
let transaction = match transactions.get(&transaction_id) {
Some(transaction) => transaction,
None => bail!("The transaction '{transaction_id}' for commitment '{commitment}' is not in the block"),
};
let transaction_leaf = transaction.to_leaf(transition.id())?;
let transaction_path = transaction.to_path(&transaction_leaf)?;
let block_header = block.header();
let header_root = block_header.to_root()?;
let header_leaf = HeaderLeaf::<N>::new(1, block_header.transactions_root());
let header_path = block_header.to_path(&header_leaf)?;
Ok(StatePath::from(
global_state_root.into(),
block_path,
block.hash(),
block.previous_hash(),
header_root,
header_path,
header_leaf,
transactions_path,
transaction.id(),
transaction_path,
transaction_leaf,
transition_root,
*transition.tcm(),
transition_path,
transition_leaf,
))
}
fn get_previous_block_hash(&self, height: u32) -> Result<Option<N::BlockHash>> {
match height.is_zero() {
true => Ok(Some(N::BlockHash::default())),
false => match self.id_map().get_confirmed(&(height - 1))? {
Some(block_hash) => Ok(Some(cow_to_copied!(block_hash))),
None => Ok(None),
},
}
}
fn get_block_hash(&self, height: u32) -> Result<Option<N::BlockHash>> {
match self.id_map().get_confirmed(&height)? {
Some(block_hash) => Ok(Some(cow_to_copied!(block_hash))),
None => Ok(None),
}
}
fn get_block_height(&self, block_hash: &N::BlockHash) -> Result<Option<u32>> {
match self.reverse_id_map().get_confirmed(block_hash)? {
Some(height) => Ok(Some(cow_to_copied!(height))),
None => Ok(None),
}
}
fn get_block_header(&self, block_hash: &N::BlockHash) -> Result<Option<Header<N>>> {
match self.header_map().get_confirmed(block_hash)? {
Some(header) => Ok(Some(cow_to_cloned!(header))),
None => Ok(None),
}
}
fn get_block_authority(&self, block_hash: &N::BlockHash) -> Result<Option<Authority<N>>> {
match self.authority_map().get_confirmed(block_hash)? {
Some(authority) => Ok(Some(cow_to_cloned!(authority))),
None => Ok(None),
}
}
fn get_batch_certificate(&self, certificate_id: &Field<N>) -> Result<Option<BatchCertificate<N>>> {
let (block_height, round) = match self.certificate_map().get_confirmed(certificate_id)? {
Some(block_height_and_round) => cow_to_copied!(block_height_and_round),
None => return Ok(None),
};
let Some(block_hash) = self.get_block_hash(block_height)? else {
bail!("The block hash for block '{block_height}' is missing in block storage")
};
let Some(authority) = self.authority_map().get_confirmed(&block_hash)? else {
bail!("The authority for '{block_hash}' is missing in block storage")
};
match authority {
Cow::Owned(Authority::Quorum(ref subdag)) | Cow::Borrowed(Authority::Quorum(ref subdag)) => {
match subdag.get(&round) {
Some(certificates) => {
match certificates.iter().find(|certificate| &certificate.id() == certificate_id) {
Some(certificate) => Ok(Some(certificate.clone())),
None => bail!("The certificate '{certificate_id}' is missing in block storage"),
}
}
None => bail!("The certificates for round '{round}' is missing in block storage"),
}
}
_ => bail!(
"Cannot fetch batch certificate '{certificate_id}' - The authority for block '{block_height}' is not a subdag"
),
}
}
fn get_block_ratifications(&self, block_hash: &N::BlockHash) -> Result<Option<Ratifications<N>>> {
match self.ratifications_map().get_confirmed(block_hash)? {
Some(ratifications) => Ok(Some(cow_to_cloned!(ratifications))),
None => Ok(None),
}
}
fn get_block_solutions(&self, block_hash: &N::BlockHash) -> Result<Option<CoinbaseSolution<N>>> {
match self.solutions_map().get_confirmed(block_hash)? {
Some(solutions) => Ok(cow_to_cloned!(solutions)),
None => bail!("Missing solutions for block ('{block_hash}')"),
}
}
fn get_solution(&self, puzzle_commitment: &PuzzleCommitment<N>) -> Result<ProverSolution<N>> {
let Some(block_height) = self.find_block_height_from_puzzle_commitment(puzzle_commitment)? else {
bail!("The block height for puzzle commitment '{puzzle_commitment}' is missing in block storage")
};
let Some(block_hash) = self.get_block_hash(block_height)? else {
bail!("The block hash for block '{block_height}' is missing in block storage")
};
let Some(solutions) = self.solutions_map().get_confirmed(&block_hash)? else {
bail!("The solutions for block '{block_height}' are missing in block storage")
};
match solutions {
Cow::Owned(Some(ref solutions)) | Cow::Borrowed(Some(ref solutions)) => {
solutions.get(puzzle_commitment).cloned().ok_or_else(|| {
anyhow!(
"The prover solution for puzzle commitment '{puzzle_commitment}' is missing in block storage"
)
})
}
_ => bail!("The prover solution for puzzle commitment '{puzzle_commitment}' is missing in block storage"),
}
}
fn get_block_transactions(&self, block_hash: &N::BlockHash) -> Result<Option<Transactions<N>>> {
let transaction_ids = match self.transactions_map().get_confirmed(block_hash)? {
Some(transaction_ids) => transaction_ids,
None => return Ok(None),
};
transaction_ids
.iter()
.map(|transaction_id| self.get_confirmed_transaction(*transaction_id))
.collect::<Result<Option<Transactions<_>>>>()
}
fn get_block_aborted_transaction_ids(&self, block_hash: &N::BlockHash) -> Result<Option<Vec<N::TransactionID>>> {
match self.aborted_transaction_ids_map().get_confirmed(block_hash)? {
Some(aborted_transaction_ids) => Ok(Some(cow_to_cloned!(aborted_transaction_ids))),
None => Ok(None),
}
}
fn get_transaction(&self, transaction_id: &N::TransactionID) -> Result<Option<Transaction<N>>> {
match self.rejected_or_aborted_transaction_id_map().get_confirmed(transaction_id)? {
Some(block_hash) => match self.get_block_transactions(&block_hash)? {
Some(transactions) => {
match transactions.find_confirmed_transaction_for_unconfirmed_transaction_id(transaction_id) {
Some(confirmed) => Ok(Some(confirmed.transaction().clone())),
None => bail!("Missing transaction '{transaction_id}' in block storage"),
}
}
None => bail!("Missing transactions for block '{block_hash}' in block storage"),
},
None => self.transaction_store().get_transaction(transaction_id),
}
}
fn get_confirmed_transaction(&self, transaction_id: N::TransactionID) -> Result<Option<ConfirmedTransaction<N>>> {
let transaction = match self.get_transaction(&transaction_id) {
Ok(Some(transaction)) => transaction,
Ok(None) => bail!("Missing transaction '{transaction_id}' in block storage"),
Err(err) => return Err(err),
};
let (_, confirmed_type, blob) = match self.confirmed_transactions_map().get_confirmed(&transaction.id())? {
Some(confirmed_attributes) => cow_to_cloned!(confirmed_attributes),
None => bail!("Missing confirmed transaction '{transaction_id}' in block storage"),
};
to_confirmed_transaction(confirmed_type, transaction, blob).map(Some)
}
fn get_unconfirmed_transaction(&self, transaction_id: &N::TransactionID) -> Result<Option<Transaction<N>>> {
match self.rejected_or_aborted_transaction_id_map().get_confirmed(transaction_id)? {
Some(block_hash) => match self.get_block_transactions(&block_hash)? {
Some(transactions) => {
match transactions.find_confirmed_transaction_for_unconfirmed_transaction_id(transaction_id) {
Some(confirmed) => Ok(Some(confirmed.to_unconfirmed_transaction()?)),
None => bail!("Missing transaction '{transaction_id}' in block storage"),
}
}
None => bail!("Missing transactions for block '{block_hash}' in block storage"),
},
None => self.transaction_store().get_transaction(transaction_id),
}
}
fn get_block(&self, block_hash: &N::BlockHash) -> Result<Option<Block<N>>> {
let Some(height) = self.get_block_height(block_hash)? else { return Ok(None) };
let Some(header) = self.get_block_header(block_hash)? else {
bail!("Missing block header for block {height} ('{block_hash}')");
};
if header.height() != height {
bail!("Mismatching block height for block {height} ('{block_hash}')")
}
let Some(previous_hash) = self.get_previous_block_hash(height)? else {
bail!("Missing previous block hash for block {height} ('{block_hash}')");
};
let Some(authority) = self.get_block_authority(block_hash)? else {
bail!("Missing authority for block {height} ('{block_hash}')");
};
let Some(ratifications) = self.get_block_ratifications(block_hash)? else {
bail!("Missing ratifications for block {height} ('{block_hash}')");
};
let Ok(solutions) = self.get_block_solutions(block_hash) else {
bail!("Missing solutions for block {height} ('{block_hash}')");
};
let Some(transactions) = self.get_block_transactions(block_hash)? else {
bail!("Missing transactions for block {height} ('{block_hash}')");
};
let Some(aborted_transaction_ids) = self.get_block_aborted_transaction_ids(block_hash)? else {
bail!("Missing aborted transaction IDs for block {height} ('{block_hash}')");
};
Ok(Some(Block::from(
previous_hash,
header,
authority,
ratifications,
solutions,
transactions,
aborted_transaction_ids,
)?))
}
}
#[derive(Clone)]
pub struct BlockStore<N: Network, B: BlockStorage<N>> {
storage: B,
tree: Arc<RwLock<BlockTree<N>>>,
}
impl<N: Network, B: BlockStorage<N>> BlockStore<N, B> {
pub fn open(dev: Option<u16>) -> Result<Self> {
let storage = B::open(dev)?;
let tree = {
let heights = storage.id_map().keys_confirmed();
let hashes = match heights.max() {
Some(height) => cfg_into_iter!(0..=cow_to_copied!(height))
.map(|height| match storage.get_block_hash(height)? {
Some(hash) => Ok(hash.to_bits_le()),
None => bail!("Missing block hash for block {height}"),
})
.collect::<Result<Vec<Vec<bool>>>>()?,
None => vec![],
};
Arc::new(RwLock::new(N::merkle_tree_bhp(&hashes)?))
};
Ok(Self { storage, tree })
}
pub fn insert(&self, block: &Block<N>) -> Result<()> {
let mut tree = self.tree.write();
let updated_tree = tree.prepare_append(&[block.hash().to_bits_le()])?;
if block.height() != u32::try_from(updated_tree.number_of_leaves())? - 1 {
bail!("Attempted to insert a block at the incorrect height into storage")
}
self.storage.insert((*updated_tree.root()).into(), block)?;
*tree = updated_tree;
Ok(())
}
pub fn remove_last_n(&self, n: u32) -> Result<()> {
ensure!(n > 0, "Cannot remove zero blocks");
let mut tree = self.tree.write();
let heights = match self.storage.id_map().keys_confirmed().max() {
Some(height) => {
let end_height = cow_to_copied!(height);
let start_height = end_height
.checked_sub(n - 1)
.ok_or_else(|| anyhow!("Failed to remove last '{n}' blocks: block height underflow"))?;
ensure!(end_height == u32::try_from(tree.number_of_leaves())? - 1, "Block height mismatch");
start_height..=end_height
}
None => bail!("Failed to remove last '{n}' blocks: no blocks in storage"),
};
let hashes = cfg_into_iter!(heights)
.map(|height| match self.storage.get_block_hash(height)? {
Some(hash) => Ok(hash),
None => bail!("Failed to remove last '{n}' blocks: missing block hash for block {height}"),
})
.collect::<Result<Vec<_>>>()?;
let updated_tree = tree.prepare_remove_last_n(usize::try_from(n)?)?;
atomic_batch_scope!(self, {
for block_hash in hashes.iter().rev() {
self.storage.remove(block_hash)?;
}
Ok(())
})?;
*tree = updated_tree;
Ok(())
}
pub fn transaction_store(&self) -> &TransactionStore<N, B::TransactionStorage> {
self.storage.transaction_store()
}
pub fn transition_store(&self) -> &TransitionStore<N, B::TransitionStorage> {
self.storage.transaction_store().transition_store()
}
pub fn start_atomic(&self) {
self.storage.start_atomic();
}
pub fn is_atomic_in_progress(&self) -> bool {
self.storage.is_atomic_in_progress()
}
pub fn atomic_checkpoint(&self) {
self.storage.atomic_checkpoint();
}
pub fn clear_latest_checkpoint(&self) {
self.storage.clear_latest_checkpoint();
}
pub fn atomic_rewind(&self) {
self.storage.atomic_rewind();
}
pub fn abort_atomic(&self) {
self.storage.abort_atomic();
}
pub fn finish_atomic(&self) -> Result<()> {
self.storage.finish_atomic()
}
pub fn dev(&self) -> Option<u16> {
self.storage.dev()
}
}
impl<N: Network, B: BlockStorage<N>> BlockStore<N, B> {
pub fn find_block_height_from_state_root(&self, state_root: N::StateRoot) -> Result<Option<u32>> {
self.storage.find_block_height_from_state_root(state_root)
}
pub fn find_block_hash(&self, transaction_id: &N::TransactionID) -> Result<Option<N::BlockHash>> {
self.storage.find_block_hash(transaction_id)
}
pub fn find_block_height_from_puzzle_commitment(
&self,
puzzle_commitment: &PuzzleCommitment<N>,
) -> Result<Option<u32>> {
self.storage.find_block_height_from_puzzle_commitment(puzzle_commitment)
}
}
impl<N: Network, B: BlockStorage<N>> BlockStore<N, B> {
pub fn current_state_root(&self) -> N::StateRoot {
(*self.tree.read().root()).into()
}
pub fn get_state_root(&self, block_height: u32) -> Result<Option<N::StateRoot>> {
self.storage.get_state_root(block_height)
}
pub fn get_state_path_for_commitment(&self, commitment: &Field<N>) -> Result<StatePath<N>> {
self.storage.get_state_path_for_commitment(commitment, &self.tree.read())
}
pub fn get_previous_block_hash(&self, height: u32) -> Result<Option<N::BlockHash>> {
self.storage.get_previous_block_hash(height)
}
pub fn get_block_hash(&self, height: u32) -> Result<Option<N::BlockHash>> {
self.storage.get_block_hash(height)
}
pub fn get_block_height(&self, block_hash: &N::BlockHash) -> Result<Option<u32>> {
self.storage.get_block_height(block_hash)
}
pub fn get_block_header(&self, block_hash: &N::BlockHash) -> Result<Option<Header<N>>> {
self.storage.get_block_header(block_hash)
}
pub fn get_block_authority(&self, block_hash: &N::BlockHash) -> Result<Option<Authority<N>>> {
self.storage.get_block_authority(block_hash)
}
pub fn get_block_ratifications(&self, block_hash: &N::BlockHash) -> Result<Option<Ratifications<N>>> {
self.storage.get_block_ratifications(block_hash)
}
pub fn get_block_solutions(&self, block_hash: &N::BlockHash) -> Result<Option<CoinbaseSolution<N>>> {
self.storage.get_block_solutions(block_hash)
}
pub fn get_solution(&self, solution_id: &PuzzleCommitment<N>) -> Result<ProverSolution<N>> {
self.storage.get_solution(solution_id)
}
pub fn get_block_transactions(&self, block_hash: &N::BlockHash) -> Result<Option<Transactions<N>>> {
self.storage.get_block_transactions(block_hash)
}
pub fn get_block_aborted_transaction_ids(
&self,
block_hash: &N::BlockHash,
) -> Result<Option<Vec<N::TransactionID>>> {
self.storage.get_block_aborted_transaction_ids(block_hash)
}
pub fn get_transaction(&self, transaction_id: &N::TransactionID) -> Result<Option<Transaction<N>>> {
self.storage.get_transaction(transaction_id)
}
pub fn get_confirmed_transaction(
&self,
transaction_id: &N::TransactionID,
) -> Result<Option<ConfirmedTransaction<N>>> {
self.storage.get_confirmed_transaction(*transaction_id)
}
pub fn get_unconfirmed_transaction(&self, transaction_id: &N::TransactionID) -> Result<Option<Transaction<N>>> {
self.storage.get_unconfirmed_transaction(transaction_id)
}
pub fn get_block(&self, block_hash: &N::BlockHash) -> Result<Option<Block<N>>> {
self.storage.get_block(block_hash)
}
pub fn get_program(&self, program_id: &ProgramID<N>) -> Result<Option<Program<N>>> {
self.storage.transaction_store().get_program(program_id)
}
pub fn get_batch_certificate(&self, certificate_id: &Field<N>) -> Result<Option<BatchCertificate<N>>> {
self.storage.get_batch_certificate(certificate_id)
}
}
impl<N: Network, B: BlockStorage<N>> BlockStore<N, B> {
pub fn contains_state_root(&self, state_root: &N::StateRoot) -> Result<bool> {
self.storage.reverse_state_root_map().contains_key_confirmed(state_root)
}
pub fn contains_block_height(&self, height: u32) -> Result<bool> {
self.storage.id_map().contains_key_confirmed(&height)
}
pub fn contains_block_hash(&self, block_hash: &N::BlockHash) -> Result<bool> {
self.storage.reverse_id_map().contains_key_confirmed(block_hash)
}
pub fn contains_transaction_id(&self, transaction_id: &N::TransactionID) -> Result<bool> {
self.storage.contains_transaction_id(transaction_id)
}
pub fn contains_rejected_or_aborted_transaction_id(&self, transaction_id: &N::TransactionID) -> Result<bool> {
self.storage.contains_rejected_or_aborted_transaction_id(transaction_id)
}
pub fn contains_rejected_deployment_or_execution_id(&self, rejected_id: &Field<N>) -> Result<bool> {
self.storage.contains_rejected_deployment_or_execution_id(rejected_id)
}
pub fn contains_certificate(&self, certificate_id: &Field<N>) -> Result<bool> {
self.storage.certificate_map().contains_key_confirmed(certificate_id)
}
pub fn contains_puzzle_commitment(&self, puzzle_commitment: &PuzzleCommitment<N>) -> Result<bool> {
self.storage.puzzle_commitments_map().contains_key_confirmed(puzzle_commitment)
}
}
impl<N: Network, B: BlockStorage<N>> BlockStore<N, B> {
pub fn state_roots(&self) -> impl '_ + Iterator<Item = Cow<'_, N::StateRoot>> {
self.storage.reverse_state_root_map().keys_confirmed()
}
pub fn heights(&self) -> impl '_ + Iterator<Item = Cow<'_, u32>> {
self.storage.id_map().keys_confirmed()
}
pub fn hashes(&self) -> impl '_ + Iterator<Item = Cow<'_, N::BlockHash>> {
self.storage.reverse_id_map().keys_confirmed()
}
pub fn puzzle_commitments(&self) -> impl '_ + Iterator<Item = Cow<'_, PuzzleCommitment<N>>> {
self.storage.puzzle_commitments_map().keys_confirmed()
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::helpers::memory::BlockMemory;
type CurrentNetwork = console::network::Testnet3;
#[test]
fn test_insert_get_remove() {
let rng = &mut TestRng::default();
let block = ledger_test_helpers::sample_genesis_block(rng);
let block_hash = block.hash();
let block_store = BlockStore::<CurrentNetwork, BlockMemory<_>>::open(None).unwrap();
let candidate = block_store.get_block(&block_hash).unwrap();
assert_eq!(None, candidate);
block_store.insert(&block).unwrap();
let candidate = block_store.get_block(&block_hash).unwrap();
assert_eq!(Some(block), candidate);
block_store.remove_last_n(1).unwrap();
let candidate = block_store.get_block(&block_hash).unwrap();
assert_eq!(None, candidate);
}
#[test]
fn test_find_block_hash() {
let rng = &mut TestRng::default();
let block = ledger_test_helpers::sample_genesis_block(rng);
let block_hash = block.hash();
assert!(block.transactions().num_accepted() > 0, "This test must be run with at least one transaction.");
let block_store = BlockStore::<CurrentNetwork, BlockMemory<_>>::open(None).unwrap();
let candidate = block_store.get_block(&block_hash).unwrap();
assert_eq!(None, candidate);
for transaction_id in block.transaction_ids() {
let candidate = block_store.find_block_hash(transaction_id).unwrap();
assert_eq!(None, candidate);
}
block_store.insert(&block).unwrap();
for transaction_id in block.transaction_ids() {
let candidate = block_store.find_block_hash(transaction_id).unwrap();
assert_eq!(Some(block_hash), candidate);
}
block_store.remove_last_n(1).unwrap();
for transaction_id in block.transaction_ids() {
let candidate = block_store.find_block_hash(transaction_id).unwrap();
assert_eq!(None, candidate);
}
}
#[test]
fn test_get_transaction() {
let rng = &mut TestRng::default();
let block = ledger_test_helpers::sample_genesis_block(rng);
assert!(block.transactions().num_accepted() > 0, "This test must be run with at least one transaction.");
let block_store = BlockStore::<CurrentNetwork, BlockMemory<_>>::open(None).unwrap();
block_store.insert(&block).unwrap();
for confirmed in block.transactions().clone().into_iter() {
assert_eq!(block_store.get_transaction(&confirmed.id()).unwrap().unwrap(), confirmed.into_transaction());
}
}
#[test]
fn test_get_confirmed_transaction() {
let rng = &mut TestRng::default();
let block = ledger_test_helpers::sample_genesis_block(rng);
assert!(block.transactions().num_accepted() > 0, "This test must be run with at least one transaction.");
let block_store = BlockStore::<CurrentNetwork, BlockMemory<_>>::open(None).unwrap();
block_store.insert(&block).unwrap();
for confirmed in block.transactions().clone().into_iter() {
assert_eq!(block_store.get_confirmed_transaction(&confirmed.id()).unwrap().unwrap(), confirmed);
}
}
#[test]
fn test_get_unconfirmed_transaction() {
let rng = &mut TestRng::default();
let block = ledger_test_helpers::sample_genesis_block(rng);
assert!(block.transactions().num_accepted() > 0, "This test must be run with at least one transaction.");
let block_store = BlockStore::<CurrentNetwork, BlockMemory<_>>::open(None).unwrap();
block_store.insert(&block).unwrap();
for confirmed in block.transactions().clone().into_iter() {
assert_eq!(
block_store.get_unconfirmed_transaction(&confirmed.id()).unwrap().unwrap(),
confirmed.to_unconfirmed_transaction().unwrap()
);
}
}
}