From 2d44c0cf379a2aa5270362f37cab81298248e32d Mon Sep 17 00:00:00 2001 From: Ardit Marku Date: Mon, 23 Dec 2024 18:48:33 +0200 Subject: [PATCH 1/2] Include human-friendly error messages on EVM transaction events --- fvm/evm/events/events.go | 12 ++- fvm/evm/evm_test.go | 168 +++++++++++++++++++++++++++++++++++++++ 2 files changed, 179 insertions(+), 1 deletion(-) diff --git a/fvm/evm/events/events.go b/fvm/evm/events/events.go index 584f6814104..c2c9427aa6f 100644 --- a/fvm/evm/events/events.go +++ b/fvm/evm/events/events.go @@ -5,7 +5,9 @@ import ( "github.com/onflow/cadence" "github.com/onflow/cadence/encoding/ccf" + "github.com/onflow/go-ethereum/accounts/abi" gethCommon "github.com/onflow/go-ethereum/common" + gethVM "github.com/onflow/go-ethereum/core/vm" "github.com/onflow/flow-go/fvm/evm/stdlib" "github.com/onflow/flow-go/fvm/evm/types" @@ -62,13 +64,21 @@ func (p *transactionEvent) ToCadence(chainID flow.ChainID) (cadence.Event, error eventType := stdlib.CadenceTypesForChain(chainID).TransactionExecuted + errorMessage := p.Result.ErrorMsg() + if p.Result.ResultSummary().ErrorCode == types.ExecutionErrCodeExecutionReverted { + reason, errUnpack := abi.UnpackRevert(p.Result.ReturnedData) + if errUnpack == nil { + errorMessage = fmt.Sprintf("%v: %v", gethVM.ErrExecutionReverted.Error(), reason) + } + } + return cadence.NewEvent([]cadence.Value{ hashToCadenceArrayValue(p.Result.TxHash), cadence.NewUInt16(p.Result.Index), cadence.NewUInt8(p.Result.TxType), bytesToCadenceUInt8ArrayValue(p.Payload), cadence.NewUInt16(uint16(p.Result.ResultSummary().ErrorCode)), - cadence.String(p.Result.ErrorMsg()), + cadence.String(errorMessage), cadence.NewUInt64(p.Result.GasConsumed), cadence.String(p.Result.DeployedContractAddressString()), bytesToCadenceUInt8ArrayValue(encodedLogs), diff --git a/fvm/evm/evm_test.go b/fvm/evm/evm_test.go index 70357202f74..4857b532a33 100644 --- a/fvm/evm/evm_test.go +++ b/fvm/evm/evm_test.go @@ -391,6 +391,174 @@ func TestEVMRun(t *testing.T) { assert.Equal(t, num, last.Big().Int64()) }) }) + + t.Run("testing EVM.run execution reverted with assert error", func(t *testing.T) { + + t.Parallel() + + RunWithNewEnvironment(t, + chain, func( + ctx fvm.Context, + vm fvm.VM, + snapshot snapshot.SnapshotTree, + testContract *TestContract, + testAccount *EOATestAccount, + ) { + sc := systemcontracts.SystemContractsForChain(chain.ChainID()) + code := []byte(fmt.Sprintf( + ` + import EVM from %s + + transaction(tx: [UInt8], coinbaseBytes: [UInt8; 20]){ + prepare(account: &Account) { + let coinbase = EVM.EVMAddress(bytes: coinbaseBytes) + let res = EVM.run(tx: tx, coinbase: coinbase) + + assert(res.status == EVM.Status.failed, message: "unexpected status") + assert(res.errorCode == 306, message: "unexpected error code") + assert(res.deployedContract == nil, message: "unexpected deployed contract") + } + } + `, + sc.EVMContract.Address.HexWithPrefix(), + )) + + coinbaseAddr := types.Address{1, 2, 3} + coinbaseBalance := getEVMAccountBalance(t, ctx, vm, snapshot, coinbaseAddr) + require.Zero(t, types.BalanceToBigInt(coinbaseBalance).Uint64()) + + innerTxBytes := testAccount.PrepareSignAndEncodeTx(t, + testContract.DeployedAt.ToCommon(), + testContract.MakeCallData(t, "assertError"), + big.NewInt(0), + uint64(100_000), + big.NewInt(1), + ) + + innerTx := cadence.NewArray( + ConvertToCadence(innerTxBytes), + ).WithType(stdlib.EVMTransactionBytesCadenceType) + + coinbase := cadence.NewArray( + ConvertToCadence(coinbaseAddr.Bytes()), + ).WithType(stdlib.EVMAddressBytesCadenceType) + + tx := fvm.Transaction( + flow.NewTransactionBody(). + SetScript(code). + AddAuthorizer(sc.FlowServiceAccount.Address). + AddArgument(json.MustEncode(innerTx)). + AddArgument(json.MustEncode(coinbase)), + 0) + + state, output, err := vm.Run( + ctx, + tx, + snapshot, + ) + require.NoError(t, err) + require.NoError(t, output.Err) + require.NotEmpty(t, state.WriteSet) + + // assert event fields are correct + require.Len(t, output.Events, 2) + txEvent := output.Events[0] + txEventPayload := TxEventToPayload(t, txEvent, sc.EVMContract.Address) + require.NoError(t, err) + + assert.Equal( + t, + "execution reverted: Assert Error Message", + txEventPayload.ErrorMessage, + ) + }, + ) + }) + + t.Run("testing EVM.run execution reverted with custom error", func(t *testing.T) { + + t.Parallel() + + RunWithNewEnvironment(t, + chain, func( + ctx fvm.Context, + vm fvm.VM, + snapshot snapshot.SnapshotTree, + testContract *TestContract, + testAccount *EOATestAccount, + ) { + sc := systemcontracts.SystemContractsForChain(chain.ChainID()) + code := []byte(fmt.Sprintf( + ` + import EVM from %s + + transaction(tx: [UInt8], coinbaseBytes: [UInt8; 20]){ + prepare(account: &Account) { + let coinbase = EVM.EVMAddress(bytes: coinbaseBytes) + let res = EVM.run(tx: tx, coinbase: coinbase) + + assert(res.status == EVM.Status.failed, message: "unexpected status") + assert(res.errorCode == 306, message: "unexpected error code") + assert(res.deployedContract == nil, message: "unexpected deployed contract") + } + } + `, + sc.EVMContract.Address.HexWithPrefix(), + )) + + coinbaseAddr := types.Address{1, 2, 3} + coinbaseBalance := getEVMAccountBalance(t, ctx, vm, snapshot, coinbaseAddr) + require.Zero(t, types.BalanceToBigInt(coinbaseBalance).Uint64()) + + innerTxBytes := testAccount.PrepareSignAndEncodeTx(t, + testContract.DeployedAt.ToCommon(), + testContract.MakeCallData(t, "customError"), + big.NewInt(0), + uint64(100_000), + big.NewInt(1), + ) + + innerTx := cadence.NewArray( + ConvertToCadence(innerTxBytes), + ).WithType(stdlib.EVMTransactionBytesCadenceType) + + coinbase := cadence.NewArray( + ConvertToCadence(coinbaseAddr.Bytes()), + ).WithType(stdlib.EVMAddressBytesCadenceType) + + tx := fvm.Transaction( + flow.NewTransactionBody(). + SetScript(code). + AddAuthorizer(sc.FlowServiceAccount.Address). + AddArgument(json.MustEncode(innerTx)). + AddArgument(json.MustEncode(coinbase)), + 0) + + state, output, err := vm.Run( + ctx, + tx, + snapshot, + ) + require.NoError(t, err) + require.NoError(t, output.Err) + require.NotEmpty(t, state.WriteSet) + + // assert event fields are correct + require.Len(t, output.Events, 2) + txEvent := output.Events[0] + txEventPayload := TxEventToPayload(t, txEvent, sc.EVMContract.Address) + require.NoError(t, err) + + // Unlike assert errors, custom errors cannot be further examined + // or ABI decoded, as we do not have access to the contract's ABI. + assert.Equal( + t, + "execution reverted", + txEventPayload.ErrorMessage, + ) + }, + ) + }) } func TestEVMBatchRun(t *testing.T) { From e682cd2dc8ee35aeafadb5b179b280253f86f4c5 Mon Sep 17 00:00:00 2001 From: Ardit Marku Date: Mon, 6 Jan 2025 12:54:07 +0200 Subject: [PATCH 2/2] Move logic for parsing revert reason messages to a dedicated method on Result type --- fvm/evm/events/events.go | 12 +----------- fvm/evm/types/result.go | 20 ++++++++++++++++++++ 2 files changed, 21 insertions(+), 11 deletions(-) diff --git a/fvm/evm/events/events.go b/fvm/evm/events/events.go index c2c9427aa6f..abca975d725 100644 --- a/fvm/evm/events/events.go +++ b/fvm/evm/events/events.go @@ -5,9 +5,7 @@ import ( "github.com/onflow/cadence" "github.com/onflow/cadence/encoding/ccf" - "github.com/onflow/go-ethereum/accounts/abi" gethCommon "github.com/onflow/go-ethereum/common" - gethVM "github.com/onflow/go-ethereum/core/vm" "github.com/onflow/flow-go/fvm/evm/stdlib" "github.com/onflow/flow-go/fvm/evm/types" @@ -64,21 +62,13 @@ func (p *transactionEvent) ToCadence(chainID flow.ChainID) (cadence.Event, error eventType := stdlib.CadenceTypesForChain(chainID).TransactionExecuted - errorMessage := p.Result.ErrorMsg() - if p.Result.ResultSummary().ErrorCode == types.ExecutionErrCodeExecutionReverted { - reason, errUnpack := abi.UnpackRevert(p.Result.ReturnedData) - if errUnpack == nil { - errorMessage = fmt.Sprintf("%v: %v", gethVM.ErrExecutionReverted.Error(), reason) - } - } - return cadence.NewEvent([]cadence.Value{ hashToCadenceArrayValue(p.Result.TxHash), cadence.NewUInt16(p.Result.Index), cadence.NewUInt8(p.Result.TxType), bytesToCadenceUInt8ArrayValue(p.Payload), cadence.NewUInt16(uint16(p.Result.ResultSummary().ErrorCode)), - cadence.String(errorMessage), + cadence.String(p.Result.ErrorMessageWithRevertReason()), cadence.NewUInt64(p.Result.GasConsumed), cadence.String(p.Result.DeployedContractAddressString()), bytesToCadenceUInt8ArrayValue(encodedLogs), diff --git a/fvm/evm/types/result.go b/fvm/evm/types/result.go index 2ad16a0b5df..998692f1017 100644 --- a/fvm/evm/types/result.go +++ b/fvm/evm/types/result.go @@ -1,8 +1,12 @@ package types import ( + "fmt" + + "github.com/onflow/go-ethereum/accounts/abi" gethCommon "github.com/onflow/go-ethereum/common" gethTypes "github.com/onflow/go-ethereum/core/types" + gethVM "github.com/onflow/go-ethereum/core/vm" "github.com/onflow/go-ethereum/rlp" ) @@ -144,6 +148,22 @@ func (res *Result) ErrorMsg() string { return errorMsg } +// ErrorMessageWithRevertReason returns the error message, if any VM or Validation +// error occurred. Execution reverts coming from `assert` or `require` Solidity +// statements, are parsed into their human-friendly representation. +func (res *Result) ErrorMessageWithRevertReason() string { + errorMessage := res.ErrorMsg() + + if res.ResultSummary().ErrorCode == ExecutionErrCodeExecutionReverted { + reason, errUnpack := abi.UnpackRevert(res.ReturnedData) + if errUnpack == nil { + errorMessage = fmt.Sprintf("%v: %v", gethVM.ErrExecutionReverted.Error(), reason) + } + } + + return errorMessage +} + // RLPEncodedLogs returns the rlp encoding of the logs func (res *Result) RLPEncodedLogs() ([]byte, error) { var encodedLogs []byte