2020-09-22 14:53:52 +02:00
|
|
|
// Copyright 2015-2020 Parity Technologies (UK) Ltd.
|
|
|
|
// This file is part of OpenEthereum.
|
2016-05-16 18:33:32 +02:00
|
|
|
|
2020-09-22 14:53:52 +02:00
|
|
|
// OpenEthereum is free software: you can redistribute it and/or modify
|
2016-05-16 18:33:32 +02:00
|
|
|
// it under the terms of the GNU General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
|
2020-09-22 14:53:52 +02:00
|
|
|
// OpenEthereum is distributed in the hope that it will be useful,
|
2016-05-16 18:33:32 +02:00
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License for more details.
|
|
|
|
|
|
|
|
// You should have received a copy of the GNU General Public License
|
2020-09-22 14:53:52 +02:00
|
|
|
// along with OpenEthereum. If not, see <http://www.gnu.org/licenses/>.
|
2016-05-16 18:33:32 +02:00
|
|
|
|
|
|
|
//! Transaction execution format module.
|
|
|
|
|
2017-09-06 20:47:45 +02:00
|
|
|
use bytes::Bytes;
|
2020-08-05 06:08:03 +02:00
|
|
|
use ethereum_types::{Address, U256, U512};
|
2018-07-02 18:50:05 +02:00
|
|
|
use ethtrie;
|
2020-08-05 06:08:03 +02:00
|
|
|
use trace::{FlatTrace, VMTrace};
|
|
|
|
use types::{log_entry::LogEntry, state_diff::StateDiff};
|
2017-08-01 12:37:57 +02:00
|
|
|
use vm;
|
2016-05-16 18:33:32 +02:00
|
|
|
|
2020-08-05 06:08:03 +02:00
|
|
|
use std::{error, fmt};
|
2016-07-27 17:41:21 +02:00
|
|
|
|
2016-05-16 18:33:32 +02:00
|
|
|
/// Transaction execution receipt.
|
2016-12-21 15:09:35 +01:00
|
|
|
#[derive(Debug, PartialEq, Clone)]
|
2017-10-20 15:40:25 +02:00
|
|
|
pub struct Executed<T = FlatTrace, V = VMTrace> {
|
2020-08-05 06:08:03 +02:00
|
|
|
/// True if the outer call/create resulted in an exceptional exit.
|
|
|
|
pub exception: Option<vm::Error>,
|
|
|
|
|
|
|
|
/// Gas paid up front for execution of transaction.
|
|
|
|
pub gas: U256,
|
|
|
|
|
|
|
|
/// Gas used during execution of transaction.
|
|
|
|
pub gas_used: U256,
|
|
|
|
|
|
|
|
/// Gas refunded after the execution of transaction.
|
|
|
|
/// To get gas that was required up front, add `refunded` and `gas_used`.
|
|
|
|
pub refunded: U256,
|
|
|
|
|
|
|
|
/// Cumulative gas used in current block so far.
|
|
|
|
///
|
|
|
|
/// `cumulative_gas_used = gas_used(t0) + gas_used(t1) + ... gas_used(tn)`
|
|
|
|
///
|
|
|
|
/// where `tn` is current transaction.
|
|
|
|
pub cumulative_gas_used: U256,
|
|
|
|
|
|
|
|
/// Vector of logs generated by transaction.
|
|
|
|
pub logs: Vec<LogEntry>,
|
|
|
|
|
|
|
|
/// Addresses of contracts created during execution of transaction.
|
|
|
|
/// Ordered from earliest creation.
|
|
|
|
///
|
|
|
|
/// eg. sender creates contract A and A in constructor creates contract B
|
|
|
|
///
|
|
|
|
/// B creation ends first, and it will be the first element of the vector.
|
|
|
|
pub contracts_created: Vec<Address>,
|
|
|
|
/// Transaction output.
|
|
|
|
pub output: Bytes,
|
|
|
|
/// The trace of this transaction.
|
|
|
|
pub trace: Vec<T>,
|
|
|
|
/// The VM trace of this transaction.
|
|
|
|
pub vm_trace: Option<V>,
|
|
|
|
/// The state diff, if we traced it.
|
|
|
|
pub state_diff: Option<StateDiff>,
|
2016-05-16 18:33:32 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Result of executing the transaction.
|
2016-12-21 15:09:35 +01:00
|
|
|
#[derive(PartialEq, Debug, Clone)]
|
2016-05-16 18:33:32 +02:00
|
|
|
pub enum ExecutionError {
|
2020-08-05 06:08:03 +02:00
|
|
|
/// Returned when there gas paid for transaction execution is
|
|
|
|
/// lower than base gas required.
|
|
|
|
NotEnoughBaseGas {
|
|
|
|
/// Absolute minimum gas required.
|
|
|
|
required: U256,
|
|
|
|
/// Gas provided.
|
|
|
|
got: U256,
|
|
|
|
},
|
|
|
|
/// Returned when block (gas_used + gas) > gas_limit.
|
|
|
|
///
|
|
|
|
/// If gas =< gas_limit, upstream may try to execute the transaction
|
|
|
|
/// in next block.
|
|
|
|
BlockGasLimitReached {
|
|
|
|
/// Gas limit of block for transaction.
|
|
|
|
gas_limit: U256,
|
|
|
|
/// Gas used in block prior to transaction.
|
|
|
|
gas_used: U256,
|
|
|
|
/// Amount of gas in block.
|
|
|
|
gas: U256,
|
|
|
|
},
|
2021-06-04 12:12:24 +02:00
|
|
|
/// Transaction's max gas price is lower then block base fee.
|
|
|
|
GasPriceLowerThanBaseFee {
|
|
|
|
/// Max gas price of the transaction.
|
|
|
|
gas_price: U256,
|
|
|
|
/// Block base fee.
|
|
|
|
base_fee: U256,
|
|
|
|
},
|
2020-08-05 06:08:03 +02:00
|
|
|
/// Returned when transaction nonce does not match state nonce.
|
|
|
|
InvalidNonce {
|
|
|
|
/// Nonce expected.
|
|
|
|
expected: U256,
|
|
|
|
/// Nonce found.
|
|
|
|
got: U256,
|
|
|
|
},
|
|
|
|
/// Returned when cost of transaction (value + gas_price * gas) exceeds
|
|
|
|
/// current sender balance.
|
|
|
|
NotEnoughCash {
|
|
|
|
/// Minimum required balance.
|
|
|
|
required: U512,
|
|
|
|
/// Actual balance.
|
|
|
|
got: U512,
|
|
|
|
},
|
|
|
|
/// When execution tries to modify the state in static context
|
|
|
|
MutableCallInStaticContext,
|
|
|
|
/// Returned when transacting from a non-existing account with dust protection enabled.
|
|
|
|
SenderMustExist,
|
|
|
|
/// Returned when internal evm error occurs.
|
|
|
|
Internal(String),
|
|
|
|
/// Returned when generic transaction occurs
|
|
|
|
TransactionMalformed(String),
|
2016-05-16 18:33:32 +02:00
|
|
|
}
|
|
|
|
|
2018-07-02 18:50:05 +02:00
|
|
|
impl From<Box<ethtrie::TrieError>> for ExecutionError {
|
2020-08-05 06:08:03 +02:00
|
|
|
fn from(err: Box<ethtrie::TrieError>) -> Self {
|
|
|
|
ExecutionError::Internal(format!("{:?}", err))
|
|
|
|
}
|
2018-07-02 18:50:05 +02:00
|
|
|
}
|
|
|
|
impl From<ethtrie::TrieError> for ExecutionError {
|
2020-08-05 06:08:03 +02:00
|
|
|
fn from(err: ethtrie::TrieError) -> Self {
|
|
|
|
ExecutionError::Internal(format!("{:?}", err))
|
|
|
|
}
|
2017-02-26 13:10:50 +01:00
|
|
|
}
|
|
|
|
|
2016-05-21 00:12:51 +02:00
|
|
|
impl fmt::Display for ExecutionError {
|
2020-08-05 06:08:03 +02:00
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
use self::ExecutionError::*;
|
|
|
|
|
|
|
|
let msg = match *self {
|
|
|
|
NotEnoughBaseGas {
|
|
|
|
ref required,
|
|
|
|
ref got,
|
|
|
|
} => format!(
|
|
|
|
"Not enough base gas. {} is required, but only {} paid",
|
|
|
|
required, got
|
|
|
|
),
|
|
|
|
BlockGasLimitReached {
|
|
|
|
ref gas_limit,
|
|
|
|
ref gas_used,
|
|
|
|
ref gas,
|
|
|
|
} => format!(
|
|
|
|
"Block gas limit reached. The limit is {}, {} has \
|
|
|
|
already been used, and {} more is required",
|
|
|
|
gas_limit, gas_used, gas
|
|
|
|
),
|
2021-06-04 12:12:24 +02:00
|
|
|
GasPriceLowerThanBaseFee {
|
|
|
|
ref gas_price,
|
|
|
|
ref base_fee,
|
|
|
|
} => format!(
|
|
|
|
"Max gas price is lowert than block base fee. Gas price is {}, while base fee is {}",
|
|
|
|
gas_price, base_fee
|
|
|
|
),
|
2020-08-05 06:08:03 +02:00
|
|
|
InvalidNonce {
|
|
|
|
ref expected,
|
|
|
|
ref got,
|
|
|
|
} => format!(
|
|
|
|
"Invalid transaction nonce: expected {}, found {}",
|
|
|
|
expected, got
|
|
|
|
),
|
|
|
|
NotEnoughCash {
|
|
|
|
ref required,
|
|
|
|
ref got,
|
|
|
|
} => format!(
|
|
|
|
"Cost of transaction exceeds sender balance. {} is required \
|
|
|
|
but the sender only has {}",
|
|
|
|
required, got
|
|
|
|
),
|
|
|
|
MutableCallInStaticContext => "Mutable Call in static context".to_owned(),
|
|
|
|
SenderMustExist => "Transacting from an empty account".to_owned(),
|
|
|
|
Internal(ref msg) => msg.clone(),
|
|
|
|
TransactionMalformed(ref err) => format!("Malformed transaction: {}", err),
|
|
|
|
};
|
|
|
|
|
|
|
|
f.write_fmt(format_args!("Transaction execution error ({}).", msg))
|
|
|
|
}
|
2016-05-21 00:12:51 +02:00
|
|
|
}
|
2016-05-16 18:33:32 +02:00
|
|
|
|
2018-04-19 11:52:54 +02:00
|
|
|
impl error::Error for ExecutionError {
|
2020-08-05 06:08:03 +02:00
|
|
|
fn description(&self) -> &str {
|
|
|
|
"Transaction execution error"
|
|
|
|
}
|
2018-04-19 11:52:54 +02:00
|
|
|
}
|
|
|
|
|
2016-07-27 21:34:32 +02:00
|
|
|
/// Result of executing the transaction.
|
2016-12-21 15:09:35 +01:00
|
|
|
#[derive(PartialEq, Debug, Clone)]
|
2016-08-04 18:17:21 +02:00
|
|
|
pub enum CallError {
|
2020-08-05 06:08:03 +02:00
|
|
|
/// Couldn't find the transaction in the chain.
|
|
|
|
TransactionNotFound,
|
|
|
|
/// Couldn't find requested block's state in the chain.
|
|
|
|
StatePruned,
|
|
|
|
/// Couldn't find an amount of gas that didn't result in an exception.
|
|
|
|
Exceptional(vm::Error),
|
|
|
|
/// Corrupt state.
|
|
|
|
StateCorrupt,
|
|
|
|
/// Error executing.
|
|
|
|
Execution(ExecutionError),
|
2016-07-27 21:34:32 +02:00
|
|
|
}
|
|
|
|
|
2016-08-04 18:17:21 +02:00
|
|
|
impl From<ExecutionError> for CallError {
|
2020-08-05 06:08:03 +02:00
|
|
|
fn from(error: ExecutionError) -> Self {
|
|
|
|
CallError::Execution(error)
|
|
|
|
}
|
2016-08-04 18:17:21 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
impl fmt::Display for CallError {
|
2020-08-05 06:08:03 +02:00
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
use self::CallError::*;
|
|
|
|
|
|
|
|
let msg = match *self {
|
|
|
|
TransactionNotFound => "Transaction couldn't be found in the chain".into(),
|
|
|
|
StatePruned => "Couldn't find the transaction block's state in the chain".into(),
|
|
|
|
Exceptional(ref e) => format!("An exception ({}) happened in the execution", e),
|
|
|
|
StateCorrupt => "Stored state found to be corrupted.".into(),
|
|
|
|
Execution(ref e) => format!("{}", e),
|
|
|
|
};
|
|
|
|
|
|
|
|
f.write_fmt(format_args!("Transaction execution error ({}).", msg))
|
|
|
|
}
|
2016-07-27 21:34:32 +02:00
|
|
|
}
|
|
|
|
|
2016-05-16 18:33:32 +02:00
|
|
|
/// Transaction execution result.
|
2019-03-06 15:30:35 +01:00
|
|
|
pub type ExecutionResult = Result<Box<Executed>, ExecutionError>;
|