Skip to content

Commit

Permalink
bubble up validate transaction error message
Browse files Browse the repository at this point in the history
  • Loading branch information
jangko committed Jun 12, 2023
1 parent 8700d8b commit 10aabd8
Show file tree
Hide file tree
Showing 4 changed files with 104 additions and 131 deletions.
43 changes: 21 additions & 22 deletions nimbus/core/executor/process_transaction.nim
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@
{.push raises: [].}

import
std/strutils,
../../common/common,
../../db/accounts_cache,
../../transaction/call_evm,
Expand All @@ -19,7 +20,6 @@ import
../../vm_types,
../../evm/async/operations,
../validate,
chronicles,
chronos,
stew/results

Expand All @@ -38,18 +38,18 @@ proc commitOrRollbackDependingOnGasUsed(
vmState: BaseVMState, accTx: SavePoint,
header: BlockHeader, tx: Transaction,
gasBurned: GasInt, priorityFee: GasInt):
Result[GasInt, void] {.raises: [].} =
Result[GasInt, string] {.raises: [].} =
# Make sure that the tx does not exceed the maximum cumulative limit as
# set in the block header. Again, the eip-1559 reference does not mention
# an early stop. It would rather detect differing values for the block
# header `gasUsed` and the `vmState.cumulativeGasUsed` at a later stage.
if header.gasLimit < vmState.cumulativeGasUsed + gasBurned:
vmState.stateDB.rollback(accTx)
debug "invalid tx: block header gasLimit reached",
maxLimit = header.gasLimit,
gasUsed = vmState.cumulativeGasUsed,
addition = gasBurned
return err()
try:
vmState.stateDB.rollback(accTx)
return err("invalid tx: block header gasLimit reached. gasLimit=$1, gasUsed=$2, addition=$3" % [
$header.gasLimit, $vmState.cumulativeGasUsed, $gasBurned])
except ValueError as ex:
return err(ex.msg)
else:
# Accept transaction and collect mining fee.
vmState.stateDB.commit(accTx)
Expand All @@ -66,15 +66,12 @@ proc asyncProcessTransactionImpl(
tx: Transaction; ## Transaction to validate
sender: EthAddress; ## tx.getSender or tx.ecRecover
header: BlockHeader; ## Header for the block containing the current tx
fork: EVMFork): Future[Result[GasInt,void]]
fork: EVMFork): Future[Result[GasInt, string]]
# wildcard exception, wrapped below
{.async, gcsafe.} =
## Modelled after `https://eips.ethereum.org/EIPS/eip-1559#specification`_
## which provides a backward compatible framwork for EIP1559.

#trace "Sender", sender
#trace "txHash", rlpHash = ty.rlpHash

let
roDB = vmState.readOnlyStateDB
baseFee256 = header.eip1559BaseFee(fork)
Expand All @@ -83,18 +80,17 @@ proc asyncProcessTransactionImpl(
priorityFee = min(tx.maxPriorityFee, tx.maxFee - baseFee)

# Return failure unless explicitely set `ok()`
var res: Result[GasInt,void] = err()
var res: Result[GasInt, string] = err("")

await ifNecessaryGetAccounts(vmState, @[sender, vmState.coinbase()])
if tx.to.isSome:
await ifNecessaryGetCode(vmState, tx.to.get)

# buy gas, then the gas goes into gasMeter
if vmState.gasPool < tx.gasLimit:
debug "gas limit reached",
gasLimit = vmState.gasPool,
gasNeeded = tx.gasLimit
return
return err("gas limit reached. gasLimit=$1, gasNeeded=$2" % [
$vmState.gasPool, $tx.gasLimit])

vmState.gasPool -= tx.gasLimit

# Actually, the eip-1559 reference does not mention an early exit.
Expand All @@ -103,14 +99,17 @@ proc asyncProcessTransactionImpl(
# before leaving is crucial for some unit tests that us a direct/deep call
# of the `processTransaction()` function. So there is no `return err()`
# statement, here.
if roDB.validateTransaction(tx, sender, header.gasLimit, baseFee256, fork):
let txRes = roDB.validateTransaction(tx, sender, header.gasLimit, baseFee256, fork)
if txRes.isOk:

# Execute the transaction.
let
accTx = vmState.stateDB.beginSavepoint
gasBurned = tx.txCallEvm(sender, vmState, fork)

res = commitOrRollbackDependingOnGasUsed(vmState, accTx, header, tx, gasBurned, priorityFee)
else:
res = err(txRes.error)

if vmState.generateWitness:
vmState.stateDB.collectWitnessData()
Expand All @@ -129,7 +128,7 @@ proc asyncProcessTransaction*(
tx: Transaction; ## Transaction to validate
sender: EthAddress; ## tx.getSender or tx.ecRecover
header: BlockHeader; ## Header for the block containing the current tx
fork: EVMFork): Future[Result[GasInt,void]]
fork: EVMFork): Future[Result[GasInt,string]]
{.async, gcsafe.} =
## Process the transaction, write the results to accounts db. The function
## returns the amount of gas burned if executed.
Expand All @@ -140,7 +139,7 @@ proc asyncProcessTransaction*(
vmState: BaseVMState; ## Parent accounts environment for transaction
tx: Transaction; ## Transaction to validate
sender: EthAddress; ## tx.getSender or tx.ecRecover
header: BlockHeader): Future[Result[GasInt,void]]
header: BlockHeader): Future[Result[GasInt,string]]
{.async, gcsafe.} =
## Variant of `asyncProcessTransaction()` with `*fork* derived
## from the `vmState` argument.
Expand All @@ -152,15 +151,15 @@ proc processTransaction*(
tx: Transaction; ## Transaction to validate
sender: EthAddress; ## tx.getSender or tx.ecRecover
header: BlockHeader; ## Header for the block containing the current tx
fork: EVMFork): Result[GasInt,void]
fork: EVMFork): Result[GasInt,string]
{.gcsafe, raises: [CatchableError].} =
return waitFor(vmState.asyncProcessTransaction(tx, sender, header, fork))

proc processTransaction*(
vmState: BaseVMState; ## Parent accounts environment for transaction
tx: Transaction; ## Transaction to validate
sender: EthAddress; ## tx.getSender or tx.ecRecover
header: BlockHeader): Result[GasInt,void]
header: BlockHeader): Result[GasInt,string]
{.gcsafe, raises: [CatchableError].} =
return waitFor(vmState.asyncProcessTransaction(tx, sender, header))

Expand Down
2 changes: 1 addition & 1 deletion nimbus/core/tx_pool/tx_tasks/tx_classify.nim
Original file line number Diff line number Diff line change
Expand Up @@ -234,7 +234,7 @@ proc classifyValidatePacked*(xp: TxPoolRef;
xp.chain.limits.trgLimit
tx = item.tx.eip1559TxNormalization(xp.chain.baseFee.GasInt, fork)

roDB.validateTransaction(tx, item.sender, gasLimit, baseFee, fork)
roDB.validateTransaction(tx, item.sender, gasLimit, baseFee, fork).isOk

proc classifyPacked*(xp: TxPoolRef; gasBurned, moreBurned: GasInt): bool =
## Classifier for *packing* (i.e. adding up `gasUsed` values after executing
Expand Down
188 changes: 81 additions & 107 deletions nimbus/core/validate.nim
Original file line number Diff line number Diff line change
Expand Up @@ -9,21 +9,16 @@
# according to those terms.

import
std/[sequtils, sets, times],
std/[sequtils, sets, times, strutils],
../common/common,
../db/accounts_cache,
".."/[errors, transaction, vm_state, vm_types],
"."/[dao, eip4844, gaslimit, withdrawals],
./pow/[difficulty, header],
./pow,
chronicles,
nimcrypto/utils,
stew/[objects, results]

# chronicles stuff
when loggingEnabled or enabledLogLevel > NONE:
import
nimcrypto/utils

from stew/byteutils
import nil

Expand Down Expand Up @@ -51,31 +46,28 @@ func isGenesis(header: BlockHeader): bool =
# ------------------------------------------------------------------------------

proc validateSeal(pow: PowRef; header: BlockHeader): Result[void,string] =
let (expMixDigest, miningValue) = try:
pow.getPowDigest(header)
try:
let (expMixDigest, miningValue) = pow.getPowDigest(header)

if expMixDigest != header.mixDigest:
let
miningHash = header.getPowSpecs.miningHash
(size, cachedHash) = try: pow.getPowCacheLookup(header.blockNumber)
except KeyError: return err("Unknown block")
except CatchableError as e: return err(e.msg)
return err("mixHash mismatch. actual=$1, expected=$2," &
" blockNumber=$3, miningHash=$4, nonce=$5, difficulty=$6," &
" size=$7, cachedHash=$8" % [
$header.mixDigest, $expMixDigest, $header.blockNumber,
$miningHash, header.nonce.toHex, $header.difficulty,
$size, $cachedHash])

let value = UInt256.fromBytesBE(miningValue.data)
if value > UInt256.high div header.difficulty:
return err("mining difficulty error")

except CatchableError as err:
return err("test")

if expMixDigest != header.mixDigest:
let
miningHash = header.getPowSpecs.miningHash
(size, cachedHash) = try: pow.getPowCacheLookup(header.blockNumber)
except KeyError: return err("Unknown block")
except CatchableError as e: return err(e.msg)
debug "mixHash mismatch",
actual = header.mixDigest,
expected = expMixDigest,
blockNumber = header.blockNumber,
miningHash = miningHash,
nonce = header.nonce.toHex,
difficulty = header.difficulty,
size = size,
cachedHash = cachedHash
return err("mixHash mismatch")

let value = UInt256.fromBytesBE(miningValue.data)
if value > UInt256.high div header.difficulty:
return err("mining difficulty error")
return err(err.msg)

ok()

Expand Down Expand Up @@ -249,22 +241,19 @@ proc validateTransaction*(
sender: EthAddress; ## tx.getSender or tx.ecRecover
maxLimit: GasInt; ## gasLimit from block header
baseFee: UInt256; ## baseFee from block header
fork: EVMFork): bool =
fork: EVMFork): Result[void, string] =
let
balance = roDB.getBalance(sender)
nonce = roDB.getNonce(sender)

if tx.txType == TxEip2930 and fork < FkBerlin:
debug "invalid tx: Eip2930 Tx type detected before Berlin"
return false
return err("invalid tx: Eip2930 Tx type detected before Berlin")

if tx.txType == TxEip1559 and fork < FkLondon:
debug "invalid tx: Eip1559 Tx type detected before London"
return false
return err("invalid tx: Eip1559 Tx type detected before London")

if fork >= FkShanghai and tx.contractCreation and tx.payload.len > EIP3860_MAX_INITCODE_SIZE:
debug "invalid tx: initcode size exceeds maximum"
return false
return err("invalid tx: initcode size exceeds maximum")

# Note that the following check bears some plausibility but is _not_
# covered by the eip-1559 reference (sort of) pseudo code, for details
Expand All @@ -281,81 +270,66 @@ proc validateTransaction*(
#
# The parallel lowGasLimit.json test never triggers the case checked below
# as the paricular transaction is omitted (the txs list is just set empty.)
if maxLimit < tx.gasLimit:
debug "invalid tx: block header gasLimit exceeded",
maxLimit,
gasLimit = tx.gasLimit
return false

# ensure that the user was willing to at least pay the base fee
if tx.maxFee < baseFee.truncate(int64):
debug "invalid tx: maxFee is smaller than baseFee",
maxFee = tx.maxFee,
baseFee
return false

# The total must be the larger of the two
if tx.maxFee < tx.maxPriorityFee:
debug "invalid tx: maxFee is smaller than maPriorityFee",
maxFee = tx.maxFee,
maxPriorityFee = tx.maxPriorityFee
return false

# the signer must be able to fully afford the transaction
let gasCost = if tx.txType >= TxEip1559:
tx.gasLimit.u256 * tx.maxFee.u256
else:
tx.gasLimit.u256 * tx.gasPrice.u256

if balance < gasCost:
debug "invalid tx: not enough cash for gas",
available = balance,
require = gasCost
return false

if balance - gasCost < tx.value:
debug "invalid tx: not enough cash to send",
available=balance,
availableMinusGas=balance-gasCost,
require=tx.value
return false

if tx.gasLimit < tx.intrinsicGas(fork):
debug "invalid tx: not enough gas to perform calculation",
available=tx.gasLimit,
require=tx.intrinsicGas(fork)
return false

if tx.nonce != nonce:
debug "invalid tx: account nonce mismatch",
txNonce=tx.nonce,
accountNonce=nonce
return false

if tx.nonce == high(uint64):
debug "invalid tx: nonce at maximum"
return false

# EIP-3607 Reject transactions from senders with deployed code
# The EIP spec claims this attack never happened before
# Clients might choose to disable this rule for RPC calls like
# `eth_call` and `eth_estimateGas`
# EOA = Externally Owned Account
let codeHash = roDB.getCodeHash(sender)
if codeHash != EMPTY_SHA3:
debug "invalid tx: sender is not an EOA",
sender=sender.toHex,
codeHash=codeHash.data.toHex
return false

true
try:
if maxLimit < tx.gasLimit:
return err("invalid tx: block header gasLimit exceeded. maxLimit=$1, gasLimit=$2" % [
$maxLimit, $tx.gasLimit])

# ensure that the user was willing to at least pay the base fee
if tx.maxFee < baseFee.truncate(int64):
return err("invalid tx: maxFee is smaller than baseFee. maxFee=$1, baseFee=$2" % [
$tx.maxFee, $baseFee])

# The total must be the larger of the two
if tx.maxFee < tx.maxPriorityFee:
return err("invalid tx: maxFee is smaller than maPriorityFee. maxFee=$1, maxPriorityFee=$2" % [
$tx.maxFee, $tx.maxPriorityFee])

# the signer must be able to fully afford the transaction
let gasCost = if tx.txType >= TxEip1559:
tx.gasLimit.u256 * tx.maxFee.u256
else:
tx.gasLimit.u256 * tx.gasPrice.u256

if balance < gasCost:
return err("invalid tx: not enough cash for gas. avail=$1, require=$2" % [
$balance, $gasCost])

if balance - gasCost < tx.value:
return err("invalid tx: not enough cash to send. avail=$1, availMinusGas=$2, require=$3" % [
$balance, $(balance-gasCost), $tx.value])

if tx.gasLimit < tx.intrinsicGas(fork):
return err("invalid tx: not enough gas to perform calculation. avail=$1, require=$2" % [
$tx.gasLimit, $tx.intrinsicGas(fork)])

if tx.nonce != nonce:
return err("invalid tx: account nonce mismatch. txNonce=$1, accNonce=$2" % [
$tx.nonce, $nonce])

if tx.nonce == high(uint64):
return err("invalid tx: nonce at maximum")

# EIP-3607 Reject transactions from senders with deployed code
# The EIP spec claims this attack never happened before
# Clients might choose to disable this rule for RPC calls like
# `eth_call` and `eth_estimateGas`
# EOA = Externally Owned Account
let codeHash = roDB.getCodeHash(sender)
if codeHash != EMPTY_SHA3:
return err("invalid tx: sender is not an EOA. sender=$1, codeHash=$2" % [
sender.toHex, codeHash.data.toHex])
except CatchableError as ex:
return err(ex.msg)

ok()

proc validateTransaction*(
vmState: BaseVMState; ## Parent accounts environment for transaction
tx: Transaction; ## tx to validate
sender: EthAddress; ## tx.getSender or tx.ecRecover
header: BlockHeader; ## Header for the block containing the current tx
fork: EVMFork): bool =
fork: EVMFork): Result[void, string] =
## Variant of `validateTransaction()`
let
roDB = vmState.readOnlyStateDB
Expand Down
2 changes: 1 addition & 1 deletion tools/t8n/transition.nim
Original file line number Diff line number Diff line change
Expand Up @@ -228,7 +228,7 @@ proc exec(ctx: var TransContext,
if rc.isErr:
rejected.add RejectedTx(
index: txIndex,
error: "processTransaction failed"
error: rc.error
)
continue

Expand Down

0 comments on commit 10aabd8

Please sign in to comment.