diff --git a/fvm/evm/debug/tracer.go b/fvm/evm/debug/tracer.go index 5af0828166c..58df192073e 100644 --- a/fvm/evm/debug/tracer.go +++ b/fvm/evm/debug/tracer.go @@ -32,22 +32,26 @@ type EVMTracer interface { var _ EVMTracer = &CallTracer{} type CallTracer struct { - logger zerolog.Logger - tracer *tracers.Tracer - uploader Uploader - blockID flow.Identifier + logger zerolog.Logger + tracer *tracers.Tracer + tracerConfig []byte + uploader Uploader + blockID flow.Identifier } func NewEVMCallTracer(uploader Uploader, logger zerolog.Logger) (*CallTracer, error) { - tracer, err := tracers.DefaultDirectory.New(tracerName, &tracers.Context{}, json.RawMessage(tracerConfig)) + tracerConfig := json.RawMessage(tracerConfig) + + tracer, err := tracers.DefaultDirectory.New(tracerName, &tracers.Context{}, tracerConfig) if err != nil { return nil, err } return &CallTracer{ - logger: logger.With().Str("module", "evm-tracer").Logger(), - tracer: tracer, - uploader: uploader, + logger: logger.With().Str("module", "evm-tracer").Logger(), + tracer: tracer, + tracerConfig: tracerConfig, + uploader: uploader, }, nil } @@ -55,6 +59,12 @@ func (t *CallTracer) TxTracer() *tracers.Tracer { return NewSafeTxTracer(t) } +func (t *CallTracer) ResetTracer() error { + var err error + t.tracer, err = tracers.DefaultDirectory.New(tracerName, &tracers.Context{}, json.RawMessage(tracerConfig)) + return err +} + func (t *CallTracer) WithBlockID(id flow.Identifier) { t.blockID = id } @@ -88,6 +98,14 @@ func (t *CallTracer) Collect(txID gethCommon.Hash) { l.Error().Err(err).Msg("failed to produce trace results") } + // reset tracing to have fresh state + if err := t.ResetTracer(); err != nil { + l.Error().Err(err). + Str("traces", string(res)). + Msg("failed to reset trace") + return + } + if err = t.uploader.Upload(TraceID(txID, t.blockID), res); err != nil { l.Error().Err(err). Str("traces", string(res)). @@ -97,6 +115,7 @@ func (t *CallTracer) Collect(txID gethCommon.Hash) { l.Debug().Msg("evm traces uploaded successfully") }() + } var NopTracer = &nopTracer{} @@ -126,134 +145,155 @@ func NewSafeTxTracer(ct *CallTracer) *tracers.Tracer { Str("block-id", ct.blockID.String()). Logger() - wrapped.OnTxStart = func( - vm *tracing.VMContext, - tx *types.Transaction, - from gethCommon.Address, - ) { - defer func() { - if r := recover(); r != nil { - err, ok := r.(error) - if !ok { - err = fmt.Errorf("panic: %v", r) + if ct.tracer.OnTxStart != nil { + wrapped.OnTxStart = func( + vm *tracing.VMContext, + tx *types.Transaction, + from gethCommon.Address, + ) { + defer func() { + if r := recover(); r != nil { + err, ok := r.(error) + if !ok { + err = fmt.Errorf("panic: %v", r) + } + l.Err(err). + Stack(). + Msg("OnTxStart trace collection failed") } - l.Err(err). - Stack(). - Msg("OnTxStart trace collection failed") - } - }() - ct.tracer.OnTxStart(vm, tx, from) + }() + ct.tracer.OnTxStart(vm, tx, from) + l.Debug().Msg("tracing OnTxStart is called") + } } - wrapped.OnTxEnd = func(receipt *types.Receipt, err error) { - defer func() { - if r := recover(); r != nil { - err, ok := r.(error) - if !ok { - err = fmt.Errorf("panic: %v", r) + if ct.tracer.OnTxEnd != nil { + wrapped.OnTxEnd = func(receipt *types.Receipt, err error) { + defer func() { + if r := recover(); r != nil { + err, ok := r.(error) + if !ok { + err = fmt.Errorf("panic: %v", r) + } + l.Err(err). + Stack(). + Msg("OnTxEnd trace collection failed") } - l.Err(err). - Stack(). - Msg("OnTxEnd trace collection failed") - } - }() - ct.tracer.OnTxEnd(receipt, err) + }() + ct.tracer.OnTxEnd(receipt, err) + l.Debug().Msg("tracing OnTxEnd is called") + } } - wrapped.OnEnter = func( - depth int, - typ byte, - from, to gethCommon.Address, - input []byte, - gas uint64, - value *big.Int, - ) { - defer func() { - if r := recover(); r != nil { - err, ok := r.(error) - if !ok { - err = fmt.Errorf("panic: %v", r) + if ct.tracer.OnEnter != nil { + wrapped.OnEnter = func( + depth int, + typ byte, + from, to gethCommon.Address, + input []byte, + gas uint64, + value *big.Int, + ) { + defer func() { + if r := recover(); r != nil { + err, ok := r.(error) + if !ok { + err = fmt.Errorf("panic: %v", r) + } + l.Err(err). + Stack(). + Msg("OnEnter trace collection failed") } - l.Err(err). - Stack(). - Msg("OnEnter trace collection failed") - } - }() - ct.tracer.OnEnter(depth, typ, from, to, input, gas, value) + }() + ct.tracer.OnEnter(depth, typ, from, to, input, gas, value) + l.Debug().Int("depth", depth).Msg("tracing OnEnter is called") + } } - wrapped.OnExit = func(depth int, output []byte, gasUsed uint64, err error, reverted bool) { - defer func() { - if r := recover(); r != nil { - err, ok := r.(error) - if !ok { - err = fmt.Errorf("panic: %v", r) + if ct.tracer.OnExit != nil { + wrapped.OnExit = func(depth int, output []byte, gasUsed uint64, err error, reverted bool) { + defer func() { + if r := recover(); r != nil { + err, ok := r.(error) + if !ok { + err = fmt.Errorf("panic: %v", r) + } + l.Err(err). + Stack(). + Msg("OnExit trace collection failed") } - l.Err(err). - Stack(). - Msg("OnExit trace collection failed") - } - }() - ct.tracer.OnExit(depth, output, gasUsed, err, reverted) + }() + ct.tracer.OnExit(depth, output, gasUsed, err, reverted) + l.Debug().Int("depth", depth).Msg("tracing OnExit is called") + } } - wrapped.OnOpcode = func( - pc uint64, - op byte, - gas, cost uint64, - scope tracing.OpContext, - rData []byte, - depth int, - err error, - ) { - defer func() { - if r := recover(); r != nil { - err, ok := r.(error) - if !ok { - err = fmt.Errorf("panic: %v", r) + if ct.tracer.OnOpcode != nil { + wrapped.OnOpcode = func( + pc uint64, + op byte, + gas, cost uint64, + scope tracing.OpContext, + rData []byte, + depth int, + err error, + ) { + defer func() { + if r := recover(); r != nil { + err, ok := r.(error) + if !ok { + err = fmt.Errorf("panic: %v", r) + } + l.Err(err). + Stack(). + Msg("OnOpcode trace collection failed") } - l.Err(err). - Stack(). - Msg("OnOpcode trace collection failed") - } - }() - ct.tracer.OnOpcode(pc, op, gas, cost, scope, rData, depth, err) + }() + ct.tracer.OnOpcode(pc, op, gas, cost, scope, rData, depth, err) + l.Debug().Msg("tracing OnOpcode is called") + } } - wrapped.OnFault = func( - pc uint64, - op byte, - gas, cost uint64, - scope tracing.OpContext, - depth int, - err error) { - defer func() { - if r := recover(); r != nil { - err, ok := r.(error) - if !ok { - err = fmt.Errorf("panic: %v", r) + if ct.tracer.OnFault != nil { + wrapped.OnFault = func( + pc uint64, + op byte, + gas, cost uint64, + scope tracing.OpContext, + depth int, + err error) { + defer func() { + if r := recover(); r != nil { + err, ok := r.(error) + if !ok { + err = fmt.Errorf("panic: %v", r) + } + l.Err(err). + Stack(). + Msg("OnFault trace collection failed") } - l.Err(err). - Stack(). - Msg("OnFault trace collection failed") - } - }() - ct.tracer.OnFault(pc, op, gas, cost, scope, depth, err) + }() + ct.tracer.OnFault(pc, op, gas, cost, scope, depth, err) + l.Debug().Msg("tracing OnFault is called") + } } - wrapped.OnGasChange = func(old, new uint64, reason tracing.GasChangeReason) { - defer func() { - if r := recover(); r != nil { - err, ok := r.(error) - if !ok { - err = fmt.Errorf("panic: %v", r) + if ct.tracer.OnGasChange != nil { + wrapped.OnGasChange = func(old, new uint64, reason tracing.GasChangeReason) { + defer func() { + if r := recover(); r != nil { + err, ok := r.(error) + if !ok { + err = fmt.Errorf("panic: %v", r) + } + l.Err(err). + Stack(). + Msg("OnGasChange trace collection failed") } - l.Err(err). - Stack(). - Msg("OnGasChange trace collection failed") - } - }() - ct.tracer.OnGasChange(old, new, reason) + }() + ct.tracer.OnGasChange(old, new, reason) + l.Debug().Msg("tracing OnGasChange is called") + } } wrapped.GetResult = ct.tracer.GetResult diff --git a/fvm/evm/debug/tracer_test.go b/fvm/evm/debug/tracer_test.go index 9518bc8ff44..480d5d17ba6 100644 --- a/fvm/evm/debug/tracer_test.go +++ b/fvm/evm/debug/tracer_test.go @@ -46,6 +46,7 @@ func Test_CallTracer(t *testing.T) { tx := gethTypes.NewTransaction(nonce, to, amount, 100, big.NewInt(10), data) tr.OnTxStart(nil, tx, from) tr.OnEnter(1, byte(vm.ADD), from, to, data, 20, big.NewInt(2)) + tr.OnExit(1, nil, 10, nil, false) tr.OnTxEnd(&gethTypes.Receipt{}, nil) tr.OnExit(0, []byte{0x02}, 200, nil, false) diff --git a/fvm/evm/emulator/emulator_test.go b/fvm/evm/emulator/emulator_test.go index a9a6992c29e..e0669888d1f 100644 --- a/fvm/evm/emulator/emulator_test.go +++ b/fvm/evm/emulator/emulator_test.go @@ -1324,6 +1324,84 @@ func TestTransactionTracing(t *testing.T) { }) + t.Run("contract interaction using run transaction with proxy", func(t *testing.T) { + runWithDeployedContract(t, func(testContract *testutils.TestContract, testAccount *testutils.EOATestAccount, emu *emulator.Emulator) { + + // deploy proxy + proxyContract := testutils.GetProxyContract(t) + RunWithNewBlockView(t, emu, func(blk types.BlockView) { + call := types.NewDeployCall( + testAccount.Address(), + proxyContract.ByteCode, + math.MaxUint64, + big.NewInt(0), + testAccount.Nonce()) + + res, err := blk.DirectCall(call) + require.NoError(t, err) + require.NotNil(t, res.DeployedContractAddress) + testAccount.SetNonce(testAccount.Nonce() + 1) + proxyContract.DeployedAt = *res.DeployedContractAddress + }) + + RunWithNewBlockView(t, emu, func(blk types.BlockView) { + // set proxy contract reference the test contract + tx := testAccount.PrepareAndSignTx( + t, + proxyContract.DeployedAt.ToCommon(), + proxyContract.MakeCallData(t, "setImplementation", testContract.DeployedAt.ToCommon()), + big.NewInt(0), + 1_000_000, + big.NewInt(0), + ) + res, err := blk.RunTransaction(tx) + requireSuccessfulExecution(t, err, res) + }) + + blk, uploader, tracer := blockWithTracer(t, emu) + + var txID gethCommon.Hash + var trace json.RawMessage + + blockID := flow.Identifier{0x02} + uploaded := make(chan struct{}) + + uploader.UploadFunc = func(id string, message json.RawMessage) error { + uploaded <- struct{}{} + require.Equal(t, debug.TraceID(txID, blockID), id) + require.Equal(t, trace, message) + require.Greater(t, len(message), 0) + return nil + } + + // make call to the proxy + tx := testAccount.PrepareAndSignTx( + t, + proxyContract.DeployedAt.ToCommon(), + testContract.MakeCallData(t, "store", big.NewInt(2)), + big.NewInt(0), + 1_000_000, + big.NewInt(0), + ) + + // interact and record trace + res, err := blk.RunTransaction(tx) + requireSuccessfulExecution(t, err, res) + txID = res.TxHash + trace, err = tracer.TxTracer().GetResult() + require.NoError(t, err) + require.NotEmpty(t, trace) + tracer.WithBlockID(blockID) + + tracer.Collect(txID) + testAccount.SetNonce(testAccount.Nonce() + 1) + require.Eventuallyf(t, func() bool { + <-uploaded + return true + }, time.Second, time.Millisecond*100, "upload did not execute") + }) + + }) t.Run("contract interaction run failed transaction", func(t *testing.T) { runWithDeployedContract(t, func(testContract *testutils.TestContract, testAccount *testutils.EOATestAccount, emu *emulator.Emulator) { blk, _, tracer := blockWithTracer(t, emu) diff --git a/fvm/evm/testutils/contract.go b/fvm/evm/testutils/contract.go index e122904f796..424877c9ce2 100644 --- a/fvm/evm/testutils/contract.go +++ b/fvm/evm/testutils/contract.go @@ -52,6 +52,13 @@ func GetDummyKittyTestContract(t testing.TB) *TestContract { } } +func GetProxyContract(t testing.TB) *TestContract { + return &TestContract{ + ABI: contracts.ProxyContractABIJSON, + ByteCode: contracts.ProxyContractBytes, + } +} + func GetFactoryTestContract(t testing.TB) *TestContract { return &TestContract{ ABI: contracts.FactoryContractABIJSON, diff --git a/fvm/evm/testutils/contracts/contracts.go b/fvm/evm/testutils/contracts/contracts.go index 83c615f3296..6a06086f5f2 100644 --- a/fvm/evm/testutils/contracts/contracts.go +++ b/fvm/evm/testutils/contracts/contracts.go @@ -21,6 +21,14 @@ var DummyKittyContractBytes, _ = hex.DecodeString(dummyKittyContractBytesInHex) //go:embed dummy_kitty_abi.json var DummyKittyContractABIJSON string +//go:embed proxy_bytes.hex +var proxyContractBytesInHex string + +var ProxyContractBytes, _ = hex.DecodeString(proxyContractBytesInHex) + +//go:embed proxy_abi.json +var ProxyContractABIJSON string + //go:embed factory_bytes.hex var factoryContractBytesInHex string diff --git a/fvm/evm/testutils/contracts/proxy.sol b/fvm/evm/testutils/contracts/proxy.sol new file mode 100644 index 00000000000..5dc3f80107b --- /dev/null +++ b/fvm/evm/testutils/contracts/proxy.sol @@ -0,0 +1,25 @@ +pragma solidity ^0.8.0; + +contract Proxy { + address private _implementation; + + function setImplementation(address implementation) external { + _implementation = implementation; + } + + fallback() external payable { + address impl = _implementation; + assembly { + calldatacopy(0, 0, calldatasize()) + let result := delegatecall(gas(), impl, 0, calldatasize(), 0, 0) + returndatacopy(0, 0, returndatasize()) + switch result + case 0 { + revert(0, returndatasize()) + } + default { + return(0, returndatasize()) + } + } + } +} \ No newline at end of file diff --git a/fvm/evm/testutils/contracts/proxy_abi.json b/fvm/evm/testutils/contracts/proxy_abi.json new file mode 100644 index 00000000000..aaba81ba703 --- /dev/null +++ b/fvm/evm/testutils/contracts/proxy_abi.json @@ -0,0 +1,19 @@ +[ + { + "stateMutability": "payable", + "type": "fallback" + }, + { + "inputs": [ + { + "internalType": "address", + "name": "implementation", + "type": "address" + } + ], + "name": "setImplementation", + "outputs": [], + "stateMutability": "nonpayable", + "type": "function" + } +] \ No newline at end of file diff --git a/fvm/evm/testutils/contracts/proxy_bytes.hex b/fvm/evm/testutils/contracts/proxy_bytes.hex new file mode 100644 index 00000000000..700cc41f67d --- /dev/null +++ b/fvm/evm/testutils/contracts/proxy_bytes.hex @@ -0,0 +1 @@ +6080604052348015600e575f80fd5b5061018e8061001c5f395ff3fe608060405260043610610021575f3560e01c8063d784d4261461006557610022565b5b5f805f9054906101000a900473ffffffffffffffffffffffffffffffffffffffff169050365f80375f80365f845af43d5f803e805f8114610061573d5ff35b3d5ffd5b348015610070575f80fd5b5061008b6004803603810190610086919061012d565b61008d565b005b805f806101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff16021790555050565b5f80fd5b5f73ffffffffffffffffffffffffffffffffffffffff82169050919050565b5f6100fc826100d3565b9050919050565b61010c816100f2565b8114610116575f80fd5b50565b5f8135905061012781610103565b92915050565b5f60208284031215610142576101416100cf565b5b5f61014f84828501610119565b9150509291505056fea2646970667358221220dc83726b5fd29997fb44f2a424f0959fe05c9736efc3bf92f405182d14cdb5fb64736f6c634300081a0033 \ No newline at end of file