mod bytes;
mod serialize;
mod string;
use crate::{Transaction, rejected::Rejected};
use console::{network::prelude::*, program::FINALIZE_ID_DEPTH, types::Field};
use synthesizer_program::FinalizeOperation;
pub type NumFinalizeSize = u16;
#[derive(Clone, PartialEq, Eq)]
pub enum ConfirmedTransaction<N: Network> {
AcceptedDeploy(u32, Transaction<N>, Vec<FinalizeOperation<N>>),
AcceptedExecute(u32, Transaction<N>, Vec<FinalizeOperation<N>>),
RejectedDeploy(u32, Transaction<N>, Rejected<N>, Vec<FinalizeOperation<N>>),
RejectedExecute(u32, Transaction<N>, Rejected<N>, Vec<FinalizeOperation<N>>),
}
impl<N: Network> ConfirmedTransaction<N> {
pub fn accepted_deploy(
index: u32,
transaction: Transaction<N>,
finalize_operations: Vec<FinalizeOperation<N>>,
) -> Result<Self> {
let (program, fee) = match &transaction {
Transaction::Deploy(_, _, deployment, fee) => (deployment.program(), fee),
Transaction::Execute(..) | Transaction::Fee(..) => {
bail!("Transaction '{}' is not a deploy transaction", transaction.id())
}
};
let (num_initialize_mappings, num_update_key_values) =
finalize_operations.iter().try_fold((0, 0), |(init, update), operation| match operation {
FinalizeOperation::InitializeMapping(..) => Ok((init + 1, update)),
FinalizeOperation::UpdateKeyValue(..) => Ok((init, update + 1)),
op => {
bail!("Transaction '{}' (deploy) contains an invalid finalize operation ({op})", transaction.id())
}
})?;
{
if num_initialize_mappings + num_update_key_values != finalize_operations.len() {
bail!(
"Transaction '{}' (deploy) must contain '{}' operations",
transaction.id(),
finalize_operations.len()
);
}
if num_initialize_mappings != program.mappings().len() {
bail!(
"Transaction '{}' (deploy) must contain '{}' 'InitializeMapping' operations (found '{num_initialize_mappings}')",
transaction.id(),
program.mappings().len(),
)
}
if num_update_key_values != fee.num_finalize_operations() {
bail!(
"Transaction '{}' (deploy) must contain {} 'UpdateKeyValue' operations (found '{num_update_key_values}')",
transaction.id(),
fee.num_finalize_operations()
);
}
}
Ok(Self::AcceptedDeploy(index, transaction, finalize_operations))
}
pub fn accepted_execute(
index: u32,
transaction: Transaction<N>,
finalize_operations: Vec<FinalizeOperation<N>>,
) -> Result<Self> {
for operation in finalize_operations.iter() {
match operation {
FinalizeOperation::InsertKeyValue(..)
| FinalizeOperation::UpdateKeyValue(..)
| FinalizeOperation::RemoveKeyValue(..) => (),
FinalizeOperation::InitializeMapping(..)
| FinalizeOperation::ReplaceMapping(..)
| FinalizeOperation::RemoveMapping(..) => {
bail!("Transaction '{}' (execute) contains an invalid finalize operation type", transaction.id())
}
}
}
match transaction.is_execute() {
true => Ok(Self::AcceptedExecute(index, transaction, finalize_operations)),
false => bail!("Transaction '{}' is not an execute transaction", transaction.id()),
}
}
pub fn rejected_deploy(
index: u32,
transaction: Transaction<N>,
rejected: Rejected<N>,
finalize_operations: Vec<FinalizeOperation<N>>,
) -> Result<Self> {
ensure!(rejected.is_deployment(), "Rejected deployment is not a deployment");
for operation in finalize_operations.iter() {
match operation {
FinalizeOperation::InsertKeyValue(..)
| FinalizeOperation::UpdateKeyValue(..)
| FinalizeOperation::RemoveKeyValue(..) => (),
FinalizeOperation::InitializeMapping(..)
| FinalizeOperation::ReplaceMapping(..)
| FinalizeOperation::RemoveMapping(..) => {
bail!("Transaction '{}' (fee) contains an invalid finalize operation type", transaction.id())
}
}
}
match transaction.is_fee() {
true => Ok(Self::RejectedDeploy(index, transaction, rejected, finalize_operations)),
false => bail!("Transaction '{}' is not a fee transaction", transaction.id()),
}
}
pub fn rejected_execute(
index: u32,
transaction: Transaction<N>,
rejected: Rejected<N>,
finalize_operations: Vec<FinalizeOperation<N>>,
) -> Result<Self> {
ensure!(rejected.is_execution(), "Rejected execution is not an execution");
for operation in finalize_operations.iter() {
match operation {
FinalizeOperation::InsertKeyValue(..)
| FinalizeOperation::UpdateKeyValue(..)
| FinalizeOperation::RemoveKeyValue(..) => (),
FinalizeOperation::InitializeMapping(..)
| FinalizeOperation::ReplaceMapping(..)
| FinalizeOperation::RemoveMapping(..) => {
bail!("Transaction '{}' (fee) contains an invalid finalize operation type", transaction.id())
}
}
}
match transaction.is_fee() {
true => Ok(Self::RejectedExecute(index, transaction, rejected, finalize_operations)),
false => bail!("Transaction '{}' is not a fee transaction", transaction.id()),
}
}
}
impl<N: Network> ConfirmedTransaction<N> {
pub const fn is_accepted(&self) -> bool {
match self {
Self::AcceptedDeploy(..) | Self::AcceptedExecute(..) => true,
Self::RejectedDeploy(..) | Self::RejectedExecute(..) => false,
}
}
pub const fn is_rejected(&self) -> bool {
!self.is_accepted()
}
pub fn contains_unconfirmed_transaction_id(&self, unconfirmed_transaction_id: &N::TransactionID) -> bool {
self.to_unconfirmed_transaction_id().map_or(false, |id| &id == unconfirmed_transaction_id)
}
}
impl<N: Network> ConfirmedTransaction<N> {
pub const fn index(&self) -> u32 {
match self {
Self::AcceptedDeploy(index, ..) => *index,
Self::AcceptedExecute(index, ..) => *index,
Self::RejectedDeploy(index, ..) => *index,
Self::RejectedExecute(index, ..) => *index,
}
}
pub const fn variant(&self) -> &str {
match self {
Self::AcceptedDeploy(..) => "accepted deploy",
Self::AcceptedExecute(..) => "accepted execute",
Self::RejectedDeploy(..) => "rejected deploy",
Self::RejectedExecute(..) => "rejected execute",
}
}
pub const fn transaction(&self) -> &Transaction<N> {
match self {
Self::AcceptedDeploy(_, transaction, _) => transaction,
Self::AcceptedExecute(_, transaction, _) => transaction,
Self::RejectedDeploy(_, transaction, _, _) => transaction,
Self::RejectedExecute(_, transaction, _, _) => transaction,
}
}
pub fn into_transaction(self) -> Transaction<N> {
match self {
Self::AcceptedDeploy(_, transaction, _) => transaction,
Self::AcceptedExecute(_, transaction, _) => transaction,
Self::RejectedDeploy(_, transaction, _, _) => transaction,
Self::RejectedExecute(_, transaction, _, _) => transaction,
}
}
pub fn num_finalize(&self) -> usize {
match self {
Self::AcceptedDeploy(_, _, finalize) => finalize.len(),
Self::AcceptedExecute(_, _, finalize) => finalize.len(),
Self::RejectedDeploy(_, _, _, finalize) => finalize.len(),
Self::RejectedExecute(_, _, _, finalize) => finalize.len(),
}
}
pub const fn finalize_operations(&self) -> &Vec<FinalizeOperation<N>> {
match self {
Self::AcceptedDeploy(_, _, finalize) => finalize,
Self::AcceptedExecute(_, _, finalize) => finalize,
Self::RejectedDeploy(_, _, _, finalize) => finalize,
Self::RejectedExecute(_, _, _, finalize) => finalize,
}
}
pub fn to_finalize_id(&self) -> Result<Field<N>> {
let leaves = self.finalize_operations().iter().map(ToBits::to_bits_le).collect::<Vec<_>>();
Ok(*N::merkle_tree_bhp::<FINALIZE_ID_DEPTH>(&leaves)?.root())
}
pub fn to_rejected_id(&self) -> Result<Option<Field<N>>> {
match self {
ConfirmedTransaction::AcceptedDeploy(..) | ConfirmedTransaction::AcceptedExecute(..) => Ok(None),
ConfirmedTransaction::RejectedDeploy(_, _, rejected, _) => Ok(Some(rejected.to_id()?)),
ConfirmedTransaction::RejectedExecute(_, _, rejected, _) => Ok(Some(rejected.to_id()?)),
}
}
pub fn to_rejected(&self) -> Option<&Rejected<N>> {
match self {
ConfirmedTransaction::AcceptedDeploy(..) | ConfirmedTransaction::AcceptedExecute(..) => None,
ConfirmedTransaction::RejectedDeploy(_, _, rejected, _) => Some(rejected),
ConfirmedTransaction::RejectedExecute(_, _, rejected, _) => Some(rejected),
}
}
pub fn to_unconfirmed_transaction_id(&self) -> Result<N::TransactionID> {
match self {
Self::AcceptedDeploy(_, transaction, _) => Ok(transaction.id()),
Self::AcceptedExecute(_, transaction, _) => Ok(transaction.id()),
Self::RejectedDeploy(_, fee_transaction, rejected, _)
| Self::RejectedExecute(_, fee_transaction, rejected, _) => {
Ok(rejected.to_unconfirmed_id(&fee_transaction.fee_transition())?.into())
}
}
}
pub fn to_unconfirmed_transaction(&self) -> Result<Transaction<N>> {
match self {
Self::AcceptedDeploy(_, transaction, _) => Ok(transaction.clone()),
Self::AcceptedExecute(_, transaction, _) => Ok(transaction.clone()),
Self::RejectedDeploy(_, fee_transaction, rejected, _) => Transaction::from_deployment(
rejected
.program_owner()
.copied()
.ok_or_else(|| anyhow!("Missing program owner for rejected transaction"))?,
rejected.deployment().cloned().ok_or_else(|| anyhow!("Missing deployment for rejected transaction"))?,
fee_transaction.fee_transition().ok_or_else(|| anyhow!("Missing fee for rejected deployment"))?,
),
Self::RejectedExecute(_, fee_transaction, rejected, _) => Transaction::from_execution(
rejected.execution().cloned().ok_or_else(|| anyhow!("Missing execution for rejected transaction"))?,
fee_transaction.fee_transition(),
),
}
}
}
impl<N: Network> Deref for ConfirmedTransaction<N> {
type Target = Transaction<N>;
fn deref(&self) -> &Self::Target {
self.transaction()
}
}
#[cfg(test)]
pub mod test_helpers {
use super::*;
use console::network::MainnetV0;
type CurrentNetwork = MainnetV0;
pub(crate) fn sample_accepted_deploy(
index: u32,
is_fee_private: bool,
rng: &mut TestRng,
) -> ConfirmedTransaction<CurrentNetwork> {
let tx = crate::transaction::test_helpers::sample_deployment_transaction(is_fee_private, rng);
let finalize_operations = match is_fee_private {
true => vec![FinalizeOperation::InitializeMapping(Uniform::rand(rng))],
false => vec![
FinalizeOperation::InitializeMapping(Uniform::rand(rng)),
FinalizeOperation::UpdateKeyValue(Uniform::rand(rng), Uniform::rand(rng), Uniform::rand(rng)),
],
};
ConfirmedTransaction::accepted_deploy(index, tx, finalize_operations).unwrap()
}
pub(crate) fn sample_accepted_execute(
index: u32,
is_fee_private: bool,
rng: &mut TestRng,
) -> ConfirmedTransaction<CurrentNetwork> {
let tx = crate::transaction::test_helpers::sample_execution_transaction_with_fee(is_fee_private, rng);
ConfirmedTransaction::accepted_execute(index, tx, vec![]).unwrap()
}
pub(crate) fn sample_rejected_deploy(
index: u32,
is_fee_private: bool,
rng: &mut TestRng,
) -> ConfirmedTransaction<CurrentNetwork> {
let fee_transaction = match is_fee_private {
true => crate::transaction::test_helpers::sample_private_fee_transaction(rng),
false => crate::transaction::test_helpers::sample_fee_public_transaction(rng),
};
let rejected = crate::rejected::test_helpers::sample_rejected_deployment(is_fee_private, rng);
ConfirmedTransaction::rejected_deploy(index, fee_transaction, rejected, vec![]).unwrap()
}
pub(crate) fn sample_rejected_execute(
index: u32,
is_fee_private: bool,
rng: &mut TestRng,
) -> ConfirmedTransaction<CurrentNetwork> {
let fee_transaction = match is_fee_private {
true => crate::transaction::test_helpers::sample_private_fee_transaction(rng),
false => crate::transaction::test_helpers::sample_fee_public_transaction(rng),
};
let rejected = crate::rejected::test_helpers::sample_rejected_execution(is_fee_private, rng);
ConfirmedTransaction::rejected_execute(index, fee_transaction, rejected, vec![]).unwrap()
}
pub(crate) fn sample_confirmed_transactions() -> Vec<ConfirmedTransaction<CurrentNetwork>> {
let rng = &mut TestRng::default();
vec![
sample_accepted_deploy(0, true, rng),
sample_accepted_deploy(0, false, rng),
sample_accepted_execute(1, true, rng),
sample_accepted_execute(1, false, rng),
sample_rejected_deploy(2, true, rng),
sample_rejected_deploy(2, false, rng),
sample_rejected_execute(3, true, rng),
sample_rejected_execute(3, false, rng),
sample_accepted_deploy(Uniform::rand(rng), true, rng),
sample_accepted_deploy(Uniform::rand(rng), false, rng),
sample_accepted_execute(Uniform::rand(rng), true, rng),
sample_accepted_execute(Uniform::rand(rng), false, rng),
sample_rejected_deploy(Uniform::rand(rng), true, rng),
sample_rejected_deploy(Uniform::rand(rng), false, rng),
sample_rejected_execute(Uniform::rand(rng), true, rng),
sample_rejected_execute(Uniform::rand(rng), false, rng),
]
}
}
#[cfg(test)]
mod test {
use super::*;
use crate::transactions::confirmed::test_helpers;
type CurrentNetwork = console::network::MainnetV0;
#[test]
fn test_accepted_execute() {
let rng = &mut TestRng::default();
let index = Uniform::rand(rng);
let tx = crate::transaction::test_helpers::sample_execution_transaction_with_fee(true, rng);
let finalize_operations = vec![
FinalizeOperation::InsertKeyValue(Uniform::rand(rng), Uniform::rand(rng), Uniform::rand(rng)),
FinalizeOperation::UpdateKeyValue(Uniform::rand(rng), Uniform::rand(rng), Uniform::rand(rng)),
FinalizeOperation::RemoveKeyValue(Uniform::rand(rng), Uniform::rand(rng)),
];
let confirmed = ConfirmedTransaction::accepted_execute(index, tx.clone(), finalize_operations.clone()).unwrap();
assert_eq!(confirmed.index(), index);
assert_eq!(confirmed.transaction(), &tx);
assert_eq!(confirmed.num_finalize(), finalize_operations.len());
assert_eq!(confirmed.finalize_operations(), &finalize_operations);
let finalize_operations = vec![FinalizeOperation::InitializeMapping(Uniform::rand(rng))];
let confirmed = ConfirmedTransaction::accepted_execute(index, tx.clone(), finalize_operations);
assert!(confirmed.is_err());
let finalize_operations = vec![FinalizeOperation::RemoveMapping(Uniform::rand(rng))];
let confirmed = ConfirmedTransaction::accepted_execute(index, tx, finalize_operations);
assert!(confirmed.is_err());
}
#[test]
fn test_contains_unconfirmed_transaction_id() {
let rng = &mut TestRng::default();
let check_contains_unconfirmed_transaction_id = |confirmed: ConfirmedTransaction<CurrentNetwork>| {
let rng = &mut TestRng::default();
let unconfirmed_transaction_id = confirmed.to_unconfirmed_transaction_id().unwrap();
assert!(confirmed.contains_unconfirmed_transaction_id(&unconfirmed_transaction_id));
assert!(!confirmed.contains_unconfirmed_transaction_id(&<CurrentNetwork as Network>::TransactionID::from(
Field::rand(rng)
)));
};
let accepted_deploy = test_helpers::sample_accepted_deploy(Uniform::rand(rng), true, rng);
check_contains_unconfirmed_transaction_id(accepted_deploy);
let accepted_deploy = test_helpers::sample_accepted_deploy(Uniform::rand(rng), false, rng);
check_contains_unconfirmed_transaction_id(accepted_deploy);
let accepted_execution = test_helpers::sample_accepted_execute(Uniform::rand(rng), true, rng);
check_contains_unconfirmed_transaction_id(accepted_execution);
let accepted_execution = test_helpers::sample_accepted_execute(Uniform::rand(rng), false, rng);
check_contains_unconfirmed_transaction_id(accepted_execution);
let rejected_deploy = test_helpers::sample_rejected_deploy(Uniform::rand(rng), true, rng);
check_contains_unconfirmed_transaction_id(rejected_deploy);
let rejected_deploy = test_helpers::sample_rejected_deploy(Uniform::rand(rng), false, rng);
check_contains_unconfirmed_transaction_id(rejected_deploy);
let rejected_execution = test_helpers::sample_rejected_execute(Uniform::rand(rng), true, rng);
check_contains_unconfirmed_transaction_id(rejected_execution);
let rejected_execution = test_helpers::sample_rejected_execute(Uniform::rand(rng), false, rng);
check_contains_unconfirmed_transaction_id(rejected_execution);
}
#[test]
fn test_unconfirmed_transaction_ids() {
let rng = &mut TestRng::default();
let accepted_deploy = test_helpers::sample_accepted_deploy(Uniform::rand(rng), true, rng);
assert_eq!(accepted_deploy.to_unconfirmed_transaction_id().unwrap(), accepted_deploy.id());
let accepted_deploy = test_helpers::sample_accepted_deploy(Uniform::rand(rng), false, rng);
assert_eq!(accepted_deploy.to_unconfirmed_transaction_id().unwrap(), accepted_deploy.id());
let accepted_execution = test_helpers::sample_accepted_execute(Uniform::rand(rng), true, rng);
assert_eq!(accepted_execution.to_unconfirmed_transaction_id().unwrap(), accepted_execution.id());
let accepted_execution = test_helpers::sample_accepted_execute(Uniform::rand(rng), false, rng);
assert_eq!(accepted_execution.to_unconfirmed_transaction_id().unwrap(), accepted_execution.id());
let rejected_deploy = test_helpers::sample_rejected_deploy(Uniform::rand(rng), true, rng);
assert_ne!(rejected_deploy.to_unconfirmed_transaction_id().unwrap(), rejected_deploy.id());
let rejected_deploy = test_helpers::sample_rejected_deploy(Uniform::rand(rng), false, rng);
assert_ne!(rejected_deploy.to_unconfirmed_transaction_id().unwrap(), rejected_deploy.id());
let rejected_execution = test_helpers::sample_rejected_execute(Uniform::rand(rng), true, rng);
assert_ne!(rejected_execution.to_unconfirmed_transaction_id().unwrap(), rejected_execution.id());
let rejected_execution = test_helpers::sample_rejected_execute(Uniform::rand(rng), false, rng);
assert_ne!(rejected_execution.to_unconfirmed_transaction_id().unwrap(), rejected_execution.id());
}
#[test]
fn test_unconfirmed_transactions() {
let rng = &mut TestRng::default();
let accepted_deploy = test_helpers::sample_accepted_deploy(Uniform::rand(rng), true, rng);
assert_eq!(&accepted_deploy.to_unconfirmed_transaction().unwrap(), accepted_deploy.transaction());
let accepted_deploy = test_helpers::sample_accepted_deploy(Uniform::rand(rng), false, rng);
assert_eq!(&accepted_deploy.to_unconfirmed_transaction().unwrap(), accepted_deploy.transaction());
let accepted_execution = test_helpers::sample_accepted_execute(Uniform::rand(rng), true, rng);
assert_eq!(&accepted_execution.to_unconfirmed_transaction().unwrap(), accepted_execution.transaction());
let accepted_execution = test_helpers::sample_accepted_execute(Uniform::rand(rng), false, rng);
assert_eq!(&accepted_execution.to_unconfirmed_transaction().unwrap(), accepted_execution.transaction());
let deployment_transaction = crate::transaction::test_helpers::sample_deployment_transaction(true, rng);
let rejected = Rejected::new_deployment(
*deployment_transaction.owner().unwrap(),
deployment_transaction.deployment().unwrap().clone(),
);
let fee = Transaction::from_fee(deployment_transaction.fee_transition().unwrap()).unwrap();
let rejected_deploy = ConfirmedTransaction::rejected_deploy(Uniform::rand(rng), fee, rejected, vec![]).unwrap();
assert_eq!(rejected_deploy.to_unconfirmed_transaction_id().unwrap(), deployment_transaction.id());
assert_eq!(rejected_deploy.to_unconfirmed_transaction().unwrap(), deployment_transaction);
let deployment_transaction = crate::transaction::test_helpers::sample_deployment_transaction(false, rng);
let rejected = Rejected::new_deployment(
*deployment_transaction.owner().unwrap(),
deployment_transaction.deployment().unwrap().clone(),
);
let fee = Transaction::from_fee(deployment_transaction.fee_transition().unwrap()).unwrap();
let rejected_deploy = ConfirmedTransaction::rejected_deploy(Uniform::rand(rng), fee, rejected, vec![]).unwrap();
assert_eq!(rejected_deploy.to_unconfirmed_transaction_id().unwrap(), deployment_transaction.id());
assert_eq!(rejected_deploy.to_unconfirmed_transaction().unwrap(), deployment_transaction);
let execution_transaction = crate::transaction::test_helpers::sample_execution_transaction_with_fee(true, rng);
let rejected = Rejected::new_execution(execution_transaction.execution().unwrap().clone());
let fee = Transaction::from_fee(execution_transaction.fee_transition().unwrap()).unwrap();
let rejected_execute =
ConfirmedTransaction::rejected_execute(Uniform::rand(rng), fee, rejected, vec![]).unwrap();
assert_eq!(rejected_execute.to_unconfirmed_transaction_id().unwrap(), execution_transaction.id());
assert_eq!(rejected_execute.to_unconfirmed_transaction().unwrap(), execution_transaction);
let execution_transaction = crate::transaction::test_helpers::sample_execution_transaction_with_fee(false, rng);
let rejected = Rejected::new_execution(execution_transaction.execution().unwrap().clone());
let fee = Transaction::from_fee(execution_transaction.fee_transition().unwrap()).unwrap();
let rejected_execute =
ConfirmedTransaction::rejected_execute(Uniform::rand(rng), fee, rejected, vec![]).unwrap();
assert_eq!(rejected_execute.to_unconfirmed_transaction_id().unwrap(), execution_transaction.id());
assert_eq!(rejected_execute.to_unconfirmed_transaction().unwrap(), execution_transaction);
}
}