use crate::{
field::Maturity,
input::{
coin::{
CoinPredicate,
CoinSigned,
},
message::{
MessageCoinPredicate,
MessageCoinSigned,
MessageDataPredicate,
MessageDataSigned,
},
},
output,
policies::PolicyType,
transaction::{
consensus_parameters::{
PredicateParameters,
TxParameters,
},
field,
Executable,
},
Chargeable,
ConsensusParameters,
Input,
Output,
Transaction,
Witness,
};
use core::hash::Hash;
use fuel_types::{
canonical,
canonical::Serialize,
Address,
BlockHeight,
Bytes32,
ChainId,
};
use hashbrown::HashMap;
use itertools::Itertools;
mod error;
#[cfg(test)]
mod tests;
pub use error::ValidityError;
impl Input {
#[cfg(any(feature = "typescript", test))]
pub fn check(
&self,
index: usize,
txhash: &Bytes32,
outputs: &[Output],
witnesses: &[Witness],
predicate_params: &PredicateParameters,
recovery_cache: &mut Option<HashMap<u16, Address>>,
) -> Result<(), ValidityError> {
self.check_without_signature(index, outputs, witnesses, predicate_params)?;
self.check_signature(index, txhash, witnesses, recovery_cache)?;
Ok(())
}
pub fn check_signature(
&self,
index: usize,
txhash: &Bytes32,
witnesses: &[Witness],
recovery_cache: &mut Option<HashMap<u16, Address>>,
) -> Result<(), ValidityError> {
match self {
Self::CoinSigned(CoinSigned {
witness_index,
owner,
..
})
| Self::MessageCoinSigned(MessageCoinSigned {
witness_index,
recipient: owner,
..
})
| Self::MessageDataSigned(MessageDataSigned {
witness_index,
recipient: owner,
..
}) => {
let recover_address = || -> Result<Address, ValidityError> {
let witness = witnesses
.get(*witness_index as usize)
.ok_or(ValidityError::InputWitnessIndexBounds { index })?;
witness.recover_witness(txhash, index)
};
let recovered_address = if let Some(cache) = recovery_cache {
if let Some(recovered_address) = cache.get(witness_index) {
*recovered_address
} else {
let recovered_address = recover_address()?;
cache.insert(*witness_index, recovered_address);
recovered_address
}
} else {
recover_address()?
};
if owner != &recovered_address {
return Err(ValidityError::InputInvalidSignature { index });
}
Ok(())
}
Self::CoinPredicate(CoinPredicate {
owner, predicate, ..
})
| Self::MessageCoinPredicate(MessageCoinPredicate {
recipient: owner,
predicate,
..
})
| Self::MessageDataPredicate(MessageDataPredicate {
recipient: owner,
predicate,
..
}) if !Input::is_predicate_owner_valid(owner, &**predicate) => {
Err(ValidityError::InputPredicateOwner { index })
}
_ => Ok(()),
}
}
pub fn check_without_signature(
&self,
index: usize,
outputs: &[Output],
witnesses: &[Witness],
predicate_params: &PredicateParameters,
) -> Result<(), ValidityError> {
match self {
Self::CoinPredicate(CoinPredicate { predicate, .. })
| Self::MessageCoinPredicate(MessageCoinPredicate { predicate, .. })
| Self::MessageDataPredicate(MessageDataPredicate { predicate, .. })
if predicate.is_empty() =>
{
Err(ValidityError::InputPredicateEmpty { index })
}
Self::CoinPredicate(CoinPredicate { predicate, .. })
| Self::MessageCoinPredicate(MessageCoinPredicate { predicate, .. })
| Self::MessageDataPredicate(MessageDataPredicate { predicate, .. })
if predicate.len() as u64 > predicate_params.max_predicate_length() =>
{
Err(ValidityError::InputPredicateLength { index })
}
Self::CoinPredicate(CoinPredicate { predicate_data, .. })
| Self::MessageCoinPredicate(MessageCoinPredicate {
predicate_data, ..
})
| Self::MessageDataPredicate(MessageDataPredicate {
predicate_data, ..
}) if predicate_data.len() as u64
> predicate_params.max_predicate_data_length() =>
{
Err(ValidityError::InputPredicateDataLength { index })
}
Self::CoinSigned(CoinSigned { witness_index, .. })
| Self::MessageCoinSigned(MessageCoinSigned { witness_index, .. })
| Self::MessageDataSigned(MessageDataSigned { witness_index, .. })
if *witness_index as usize >= witnesses.len() =>
{
Err(ValidityError::InputWitnessIndexBounds { index })
}
Self::Contract { .. }
if 1 != outputs
.iter()
.filter_map(|output| match output {
Output::Contract(output::contract::Contract {
input_index,
..
}) if *input_index as usize == index => Some(()),
_ => None,
})
.count() =>
{
Err(ValidityError::InputContractAssociatedOutputContract { index })
}
Self::MessageDataSigned(MessageDataSigned { data, .. })
| Self::MessageDataPredicate(MessageDataPredicate { data, .. })
if data.is_empty()
|| data.len() as u64 > predicate_params.max_message_data_length() =>
{
Err(ValidityError::InputMessageDataLength { index })
}
_ => Ok(()),
}
}
}
impl Output {
pub fn check(&self, index: usize, inputs: &[Input]) -> Result<(), ValidityError> {
match self {
Self::Contract(output::contract::Contract { input_index, .. }) => {
match inputs.get(*input_index as usize) {
Some(Input::Contract { .. }) => Ok(()),
_ => Err(ValidityError::OutputContractInputIndex { index }),
}
}
_ => Ok(()),
}
}
}
pub trait FormatValidityChecks {
fn check(
&self,
block_height: BlockHeight,
consensus_params: &ConsensusParameters,
) -> Result<(), ValidityError> {
self.check_without_signatures(block_height, consensus_params)?;
self.check_signatures(&consensus_params.chain_id())?;
Ok(())
}
fn check_signatures(&self, chain_id: &ChainId) -> Result<(), ValidityError>;
fn check_without_signatures(
&self,
block_height: BlockHeight,
consensus_params: &ConsensusParameters,
) -> Result<(), ValidityError>;
}
impl FormatValidityChecks for Transaction {
fn check_signatures(&self, chain_id: &ChainId) -> Result<(), ValidityError> {
match self {
Self::Script(tx) => tx.check_signatures(chain_id),
Self::Create(tx) => tx.check_signatures(chain_id),
Self::Mint(tx) => tx.check_signatures(chain_id),
Self::Upgrade(tx) => tx.check_signatures(chain_id),
Self::Upload(tx) => tx.check_signatures(chain_id),
Self::Blob(tx) => tx.check_signatures(chain_id),
}
}
fn check_without_signatures(
&self,
block_height: BlockHeight,
consensus_params: &ConsensusParameters,
) -> Result<(), ValidityError> {
match self {
Self::Script(tx) => {
tx.check_without_signatures(block_height, consensus_params)
}
Self::Create(tx) => {
tx.check_without_signatures(block_height, consensus_params)
}
Self::Mint(tx) => tx.check_without_signatures(block_height, consensus_params),
Self::Upgrade(tx) => {
tx.check_without_signatures(block_height, consensus_params)
}
Self::Upload(tx) => {
tx.check_without_signatures(block_height, consensus_params)
}
Self::Blob(tx) => tx.check_without_signatures(block_height, consensus_params),
}
}
}
pub(crate) fn check_size<T>(tx: &T, tx_params: &TxParameters) -> Result<(), ValidityError>
where
T: canonical::Serialize,
{
if tx.size() as u64 > tx_params.max_size() {
Err(ValidityError::TransactionSizeLimitExceeded)?;
}
Ok(())
}
pub(crate) fn check_common_part<T>(
tx: &T,
block_height: BlockHeight,
consensus_params: &ConsensusParameters,
) -> Result<(), ValidityError>
where
T: canonical::Serialize + Chargeable + field::Outputs,
{
let tx_params = consensus_params.tx_params();
let predicate_params = consensus_params.predicate_params();
let base_asset_id = consensus_params.base_asset_id();
let gas_costs = consensus_params.gas_costs();
let fee_params = consensus_params.fee_params();
check_size(tx, tx_params)?;
if !tx.policies().is_valid() {
Err(ValidityError::TransactionPoliciesAreInvalid)?
}
if let Some(witness_limit) = tx.policies().get(PolicyType::WitnessLimit) {
let witness_size = tx.witnesses().size_dynamic();
if witness_size as u64 > witness_limit {
Err(ValidityError::TransactionWitnessLimitExceeded)?
}
}
let max_gas = tx.max_gas(gas_costs, fee_params);
if max_gas > tx_params.max_gas_per_tx() {
Err(ValidityError::TransactionMaxGasExceeded)?
}
if !tx.policies().is_set(PolicyType::MaxFee) {
Err(ValidityError::TransactionMaxFeeNotSet)?
};
if tx.maturity() > block_height {
Err(ValidityError::TransactionMaturity)?;
}
if tx.inputs().len() > tx_params.max_inputs() as usize {
Err(ValidityError::TransactionInputsMax)?
}
if tx.outputs().len() > tx_params.max_outputs() as usize {
Err(ValidityError::TransactionOutputsMax)?
}
if tx.witnesses().len() > tx_params.max_witnesses() as usize {
Err(ValidityError::TransactionWitnessesMax)?
}
let any_spendable_input = tx.inputs().iter().find(|input| match input {
Input::CoinSigned(_)
| Input::CoinPredicate(_)
| Input::MessageCoinSigned(_)
| Input::MessageCoinPredicate(_) => true,
Input::MessageDataSigned(_)
| Input::MessageDataPredicate(_)
| Input::Contract(_) => false,
});
if any_spendable_input.is_none() {
Err(ValidityError::NoSpendableInput)?
}
tx.input_asset_ids_unique(base_asset_id)
.try_for_each(|input_asset_id| {
if tx
.outputs()
.iter()
.filter_map(|output| match output {
Output::Change { asset_id, .. } if input_asset_id == asset_id => {
Some(())
}
_ => None,
})
.count()
> 1
{
return Err(ValidityError::TransactionOutputChangeAssetIdDuplicated(
*input_asset_id,
));
}
Ok(())
})?;
let duplicated_utxo_id = tx
.inputs()
.iter()
.filter_map(|i| i.is_coin().then(|| i.utxo_id()).flatten());
if let Some(utxo_id) = next_duplicate(duplicated_utxo_id).copied() {
return Err(ValidityError::DuplicateInputUtxoId { utxo_id });
}
let duplicated_contract_id = tx.inputs().iter().filter_map(Input::contract_id);
if let Some(contract_id) = next_duplicate(duplicated_contract_id).copied() {
return Err(ValidityError::DuplicateInputContractId { contract_id });
}
let duplicated_message_id = tx.inputs().iter().filter_map(Input::message_id);
if let Some(message_id) = next_duplicate(duplicated_message_id) {
return Err(ValidityError::DuplicateMessageInputId { message_id });
}
tx.inputs()
.iter()
.enumerate()
.try_for_each(|(index, input)| {
input.check_without_signature(
index,
tx.outputs(),
tx.witnesses(),
predicate_params,
)
})?;
tx.outputs()
.iter()
.enumerate()
.try_for_each(|(index, output)| {
output.check(index, tx.inputs())?;
if let Output::Change { asset_id, .. } = output {
if !tx
.input_asset_ids(base_asset_id)
.any(|input_asset_id| input_asset_id == asset_id)
{
return Err(ValidityError::TransactionOutputChangeAssetIdNotFound(
*asset_id,
));
}
}
if let Output::Coin { asset_id, .. } = output {
if !tx
.input_asset_ids(base_asset_id)
.any(|input_asset_id| input_asset_id == asset_id)
{
return Err(ValidityError::TransactionOutputCoinAssetIdNotFound(
*asset_id,
));
}
}
Ok(())
})?;
Ok(())
}
pub(crate) fn next_duplicate<U>(iter: impl Iterator<Item = U>) -> Option<U>
where
U: PartialEq + Ord + Copy + Hash,
{
#[cfg(not(feature = "std"))]
{
iter.sorted()
.as_slice()
.windows(2)
.filter_map(|u| (u[0] == u[1]).then(|| u[0]))
.next()
}
#[cfg(feature = "std")]
{
iter.duplicates().next()
}
}
#[cfg(feature = "typescript")]
mod typescript {
use crate::{
transaction::consensus_parameters::typescript::PredicateParameters,
Witness,
};
use fuel_types::Bytes32;
use wasm_bindgen::JsValue;
use alloc::{
format,
vec::Vec,
};
use crate::transaction::{
input_ts::Input,
output_ts::Output,
};
#[wasm_bindgen::prelude::wasm_bindgen]
pub fn check_input(
input: &Input,
index: usize,
txhash: &Bytes32,
outputs: Vec<JsValue>,
witnesses: Vec<JsValue>,
predicate_params: &PredicateParameters,
) -> Result<(), js_sys::Error> {
let outputs: Vec<crate::Output> = outputs
.into_iter()
.map(|v| serde_wasm_bindgen::from_value::<Output>(v).map(|v| *v.0))
.collect::<Result<Vec<_>, _>>()
.map_err(|e| js_sys::Error::new(&format!("{:?}", e)))?;
let witnesses: Vec<Witness> = witnesses
.into_iter()
.map(serde_wasm_bindgen::from_value::<Witness>)
.collect::<Result<Vec<_>, _>>()
.map_err(|e| js_sys::Error::new(&format!("{:?}", e)))?;
input
.0
.check(
index,
txhash,
&outputs,
&witnesses,
predicate_params.as_ref(),
&mut None,
)
.map_err(|e| js_sys::Error::new(&format!("{:?}", e)))
}
#[wasm_bindgen::prelude::wasm_bindgen]
pub fn check_output(
output: &Output,
index: usize,
inputs: Vec<JsValue>,
) -> Result<(), js_sys::Error> {
let inputs: Vec<crate::Input> = inputs
.into_iter()
.map(|v| serde_wasm_bindgen::from_value::<Input>(v).map(|v| *v.0))
.collect::<Result<Vec<_>, _>>()
.map_err(|e| js_sys::Error::new(&format!("{:?}", e)))?;
output
.0
.check(index, &inputs)
.map_err(|e| js_sys::Error::new(&format!("{:?}", e)))
}
}