2020-08-19 11:46:20 +00:00
|
|
|
package transactions
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2021-08-17 19:50:52 +00:00
|
|
|
"encoding/hex"
|
2020-08-19 11:46:20 +00:00
|
|
|
"errors"
|
|
|
|
"fmt"
|
2021-05-06 17:37:38 +00:00
|
|
|
"math/big"
|
2021-07-03 13:34:23 +00:00
|
|
|
"sort"
|
2020-08-19 11:46:20 +00:00
|
|
|
"time"
|
|
|
|
|
2021-05-06 17:37:38 +00:00
|
|
|
"github.com/holiman/uint256"
|
|
|
|
jsoniter "github.com/json-iterator/go"
|
2021-07-29 11:53:13 +00:00
|
|
|
"github.com/ledgerwatch/erigon-lib/kv"
|
2021-05-20 18:25:53 +00:00
|
|
|
"github.com/ledgerwatch/erigon/common"
|
|
|
|
"github.com/ledgerwatch/erigon/consensus"
|
|
|
|
"github.com/ledgerwatch/erigon/core"
|
|
|
|
"github.com/ledgerwatch/erigon/core/state"
|
|
|
|
"github.com/ledgerwatch/erigon/core/types"
|
|
|
|
"github.com/ledgerwatch/erigon/core/vm"
|
|
|
|
"github.com/ledgerwatch/erigon/eth/tracers"
|
|
|
|
"github.com/ledgerwatch/erigon/params"
|
2020-08-19 11:46:20 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
type BlockGetter interface {
|
|
|
|
// GetBlockByHash retrieves a block from the database by hash, caching it if found.
|
2020-10-24 06:57:09 +00:00
|
|
|
GetBlockByHash(hash common.Hash) (*types.Block, error)
|
2020-08-19 11:46:20 +00:00
|
|
|
// GetBlock retrieves a block from the database by hash and number,
|
|
|
|
// caching it if found.
|
|
|
|
GetBlock(hash common.Hash, number uint64) *types.Block
|
|
|
|
}
|
|
|
|
|
2022-01-24 06:47:05 +00:00
|
|
|
// ComputeTxEnv returns the execution environment of a certain transaction.
|
2022-09-01 18:49:29 +00:00
|
|
|
func ComputeTxEnv(ctx context.Context, block *types.Block, cfg *params.ChainConfig, getHeader func(hash common.Hash, number uint64) *types.Header, engine consensus.Engine, dbtx kv.Tx, blockHash common.Hash, txIndex uint64) (core.Message, vm.BlockContext, vm.TxContext, *state.IntraBlockState, *state.PlainState, error) {
|
2020-08-19 11:46:20 +00:00
|
|
|
// Create the parent state database
|
2022-02-16 08:38:12 +00:00
|
|
|
reader := state.NewPlainState(dbtx, block.NumberU64())
|
2021-04-21 02:18:05 +00:00
|
|
|
statedb := state.New(reader)
|
2020-08-19 11:46:20 +00:00
|
|
|
|
|
|
|
if txIndex == 0 && len(block.Transactions()) == 0 {
|
2021-03-14 18:52:15 +00:00
|
|
|
return nil, vm.BlockContext{}, vm.TxContext{}, statedb, reader, nil
|
2020-08-19 11:46:20 +00:00
|
|
|
}
|
|
|
|
// Recompute transactions up to the target index.
|
2021-04-22 17:11:37 +00:00
|
|
|
signer := types.MakeSigner(cfg, block.NumberU64())
|
2020-08-19 11:46:20 +00:00
|
|
|
|
2022-07-07 11:47:00 +00:00
|
|
|
header := block.Header()
|
2022-09-01 18:49:29 +00:00
|
|
|
BlockContext := core.NewEVMBlockContext(header, core.GetHashFn(header, getHeader), engine, nil)
|
2021-07-03 12:55:23 +00:00
|
|
|
vmenv := vm.NewEVM(BlockContext, vm.TxContext{}, statedb, cfg, vm.Config{})
|
2022-05-26 16:20:34 +00:00
|
|
|
rules := vmenv.ChainRules()
|
2020-08-19 11:46:20 +00:00
|
|
|
for idx, tx := range block.Transactions() {
|
|
|
|
select {
|
|
|
|
default:
|
|
|
|
case <-ctx.Done():
|
2021-03-14 18:52:15 +00:00
|
|
|
return nil, vm.BlockContext{}, vm.TxContext{}, nil, nil, ctx.Err()
|
2020-08-19 11:46:20 +00:00
|
|
|
}
|
|
|
|
statedb.Prepare(tx.Hash(), blockHash, idx)
|
|
|
|
|
|
|
|
// Assemble the transaction call message and return if the requested offset
|
2022-05-26 16:20:34 +00:00
|
|
|
msg, _ := tx.AsMessage(*signer, block.BaseFee(), rules)
|
2021-03-14 18:52:15 +00:00
|
|
|
TxContext := core.NewEVMTxContext(msg)
|
2020-08-19 11:46:20 +00:00
|
|
|
if idx == int(txIndex) {
|
2021-03-14 18:52:15 +00:00
|
|
|
return msg, BlockContext, TxContext, statedb, reader, nil
|
2020-08-19 11:46:20 +00:00
|
|
|
}
|
2021-07-03 12:55:23 +00:00
|
|
|
vmenv.Reset(TxContext, statedb)
|
2020-08-19 11:46:20 +00:00
|
|
|
// Not yet the searched for transaction, execute on top of the current state
|
2021-04-22 17:11:37 +00:00
|
|
|
if _, err := core.ApplyMessage(vmenv, msg, new(core.GasPool).AddGas(tx.GetGas()), true /* refunds */, false /* gasBailout */); err != nil {
|
2021-10-04 15:16:52 +00:00
|
|
|
return nil, vm.BlockContext{}, vm.TxContext{}, nil, nil, fmt.Errorf("transaction %x failed: %w", tx.Hash(), err)
|
2020-08-19 11:46:20 +00:00
|
|
|
}
|
|
|
|
// Ensure any modifications are committed to the state
|
2022-05-26 10:08:59 +00:00
|
|
|
// Only delete empty objects if EIP161 (part of Spurious Dragon) is in effect
|
2022-05-26 16:20:34 +00:00
|
|
|
_ = statedb.FinalizeTx(rules, reader)
|
2021-12-29 03:36:41 +00:00
|
|
|
|
|
|
|
if idx+1 == len(block.Transactions()) {
|
|
|
|
// Return the state from evaluating all txs in the block, note no msg or TxContext in this case
|
|
|
|
return nil, BlockContext, vm.TxContext{}, statedb, reader, nil
|
|
|
|
}
|
2020-08-19 11:46:20 +00:00
|
|
|
}
|
2021-03-14 18:52:15 +00:00
|
|
|
return nil, vm.BlockContext{}, vm.TxContext{}, nil, nil, fmt.Errorf("transaction index %d out of range for block %x", txIndex, blockHash)
|
2020-08-19 11:46:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// TraceTx configures a new tracer according to the provided configuration, and
|
|
|
|
// executes the given message in the provided environment. The return value will
|
|
|
|
// be tracer dependent.
|
2021-05-06 17:37:38 +00:00
|
|
|
func TraceTx(
|
|
|
|
ctx context.Context,
|
|
|
|
message core.Message,
|
|
|
|
blockCtx vm.BlockContext,
|
|
|
|
txCtx vm.TxContext,
|
|
|
|
ibs vm.IntraBlockState,
|
|
|
|
config *tracers.TraceConfig,
|
|
|
|
chainConfig *params.ChainConfig,
|
|
|
|
stream *jsoniter.Stream,
|
2022-09-17 06:25:27 +00:00
|
|
|
callTimeout time.Duration,
|
2021-05-06 17:37:38 +00:00
|
|
|
) error {
|
2020-08-19 11:46:20 +00:00
|
|
|
// Assemble the structured logger or the JavaScript tracer
|
|
|
|
var (
|
|
|
|
tracer vm.Tracer
|
|
|
|
err error
|
|
|
|
)
|
2021-05-06 17:37:38 +00:00
|
|
|
var streaming bool
|
2020-08-19 11:46:20 +00:00
|
|
|
switch {
|
|
|
|
case config != nil && config.Tracer != nil:
|
|
|
|
// Define a meaningful timeout of a single transaction trace
|
2021-07-15 10:25:32 +00:00
|
|
|
timeout := callTimeout
|
2020-08-19 11:46:20 +00:00
|
|
|
if config.Timeout != nil {
|
|
|
|
if timeout, err = time.ParseDuration(*config.Timeout); err != nil {
|
2021-06-16 17:24:56 +00:00
|
|
|
stream.WriteNil()
|
2021-05-06 17:37:38 +00:00
|
|
|
return err
|
2020-08-19 11:46:20 +00:00
|
|
|
}
|
|
|
|
}
|
2021-12-15 13:19:58 +00:00
|
|
|
// Construct the JavaScript tracer to execute with
|
|
|
|
if tracer, err = tracers.New(*config.Tracer, &tracers.Context{
|
|
|
|
TxHash: txCtx.TxHash,
|
|
|
|
}); err != nil {
|
2021-06-16 17:24:56 +00:00
|
|
|
stream.WriteNil()
|
2021-05-06 17:37:38 +00:00
|
|
|
return err
|
2020-08-19 11:46:20 +00:00
|
|
|
}
|
|
|
|
// Handle timeouts and RPC cancellations
|
|
|
|
deadlineCtx, cancel := context.WithTimeout(ctx, timeout)
|
|
|
|
go func() {
|
|
|
|
<-deadlineCtx.Done()
|
|
|
|
tracer.(*tracers.Tracer).Stop(errors.New("execution timeout"))
|
|
|
|
}()
|
|
|
|
defer cancel()
|
2021-05-06 17:37:38 +00:00
|
|
|
streaming = false
|
2020-08-19 11:46:20 +00:00
|
|
|
|
|
|
|
case config == nil:
|
2021-05-06 17:37:38 +00:00
|
|
|
tracer = NewJsonStreamLogger(nil, ctx, stream)
|
|
|
|
streaming = true
|
2020-08-19 11:46:20 +00:00
|
|
|
|
|
|
|
default:
|
2021-05-06 17:37:38 +00:00
|
|
|
tracer = NewJsonStreamLogger(config.LogConfig, ctx, stream)
|
|
|
|
streaming = true
|
2020-08-19 11:46:20 +00:00
|
|
|
}
|
|
|
|
// Run the transaction with tracing enabled.
|
2021-03-14 18:52:15 +00:00
|
|
|
vmenv := vm.NewEVM(blockCtx, txCtx, ibs, chainConfig, vm.Config{Debug: true, Tracer: tracer})
|
2022-08-10 12:04:13 +00:00
|
|
|
var refunds = true
|
2020-12-09 18:24:08 +00:00
|
|
|
if config != nil && config.NoRefunds != nil && *config.NoRefunds {
|
|
|
|
refunds = false
|
|
|
|
}
|
2021-05-06 17:37:38 +00:00
|
|
|
if streaming {
|
|
|
|
stream.WriteObjectStart()
|
|
|
|
stream.WriteObjectField("structLogs")
|
|
|
|
stream.WriteArrayStart()
|
|
|
|
}
|
2021-02-12 16:47:32 +00:00
|
|
|
result, err := core.ApplyMessage(vmenv, message, new(core.GasPool).AddGas(message.Gas()), refunds, false /* gasBailout */)
|
2020-08-19 11:46:20 +00:00
|
|
|
if err != nil {
|
2021-06-16 17:24:56 +00:00
|
|
|
if streaming {
|
|
|
|
stream.WriteArrayEnd()
|
|
|
|
stream.WriteObjectEnd()
|
|
|
|
}
|
2021-10-04 15:16:52 +00:00
|
|
|
return fmt.Errorf("tracing failed: %w", err)
|
2020-08-19 11:46:20 +00:00
|
|
|
}
|
|
|
|
// Depending on the tracer type, format and return the output
|
2021-05-06 17:37:38 +00:00
|
|
|
if streaming {
|
|
|
|
stream.WriteArrayEnd()
|
|
|
|
stream.WriteMore()
|
|
|
|
stream.WriteObjectField("gas")
|
|
|
|
stream.WriteUint64(result.UsedGas)
|
|
|
|
stream.WriteMore()
|
|
|
|
stream.WriteObjectField("failed")
|
|
|
|
stream.WriteBool(result.Failed())
|
|
|
|
stream.WriteMore()
|
2021-07-03 13:34:23 +00:00
|
|
|
// If the result contains a revert reason, return it.
|
|
|
|
returnVal := fmt.Sprintf("%x", result.Return())
|
|
|
|
if len(result.Revert()) > 0 {
|
|
|
|
returnVal = fmt.Sprintf("%x", result.Revert())
|
|
|
|
}
|
2021-05-06 17:37:38 +00:00
|
|
|
stream.WriteObjectField("returnValue")
|
2021-07-03 13:34:23 +00:00
|
|
|
stream.WriteString(returnVal)
|
2021-05-06 17:37:38 +00:00
|
|
|
stream.WriteObjectEnd()
|
|
|
|
} else {
|
|
|
|
if r, err1 := tracer.(*tracers.Tracer).GetResult(); err1 == nil {
|
|
|
|
stream.Write(r)
|
|
|
|
} else {
|
|
|
|
return err1
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// StructLogger is an EVM state logger and implements Tracer.
|
|
|
|
//
|
|
|
|
// StructLogger can capture state based on the given Log configuration and also keeps
|
|
|
|
// a track record of modified storage which is used in reporting snapshots of the
|
|
|
|
// contract their storage.
|
|
|
|
type JsonStreamLogger struct {
|
|
|
|
ctx context.Context
|
|
|
|
cfg vm.LogConfig
|
|
|
|
stream *jsoniter.Stream
|
2021-08-17 19:50:52 +00:00
|
|
|
hexEncodeBuf [128]byte
|
2021-05-06 17:37:38 +00:00
|
|
|
firstCapture bool
|
|
|
|
|
2021-07-03 13:34:23 +00:00
|
|
|
locations common.Hashes // For sorting
|
|
|
|
storage map[common.Address]vm.Storage
|
|
|
|
logs []vm.StructLog
|
|
|
|
output []byte //nolint
|
|
|
|
err error //nolint
|
2021-05-06 17:37:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewStructLogger returns a new logger
|
|
|
|
func NewJsonStreamLogger(cfg *vm.LogConfig, ctx context.Context, stream *jsoniter.Stream) *JsonStreamLogger {
|
|
|
|
logger := &JsonStreamLogger{
|
|
|
|
ctx: ctx,
|
|
|
|
stream: stream,
|
|
|
|
storage: make(map[common.Address]vm.Storage),
|
|
|
|
firstCapture: true,
|
|
|
|
}
|
|
|
|
if cfg != nil {
|
|
|
|
logger.cfg = *cfg
|
|
|
|
}
|
|
|
|
return logger
|
|
|
|
}
|
|
|
|
|
|
|
|
// CaptureStart implements the Tracer interface to initialize the tracing operation.
|
2021-12-15 13:19:58 +00:00
|
|
|
func (l *JsonStreamLogger) CaptureStart(env *vm.EVM, depth int, from common.Address, to common.Address, precompile bool, create bool, calltype vm.CallType, input []byte, gas uint64, value *big.Int, code []byte) {
|
2021-05-06 17:37:38 +00:00
|
|
|
}
|
2020-08-19 11:46:20 +00:00
|
|
|
|
2021-05-06 17:37:38 +00:00
|
|
|
// CaptureState logs a new structured log message and pushes it out to the environment
|
|
|
|
//
|
|
|
|
// CaptureState also tracks SLOAD/SSTORE ops to track storage change.
|
2021-12-15 13:19:58 +00:00
|
|
|
func (l *JsonStreamLogger) CaptureState(env *vm.EVM, pc uint64, op vm.OpCode, gas, cost uint64, scope *vm.ScopeContext, rData []byte, depth int, err error) {
|
|
|
|
contract := scope.Contract
|
|
|
|
memory := scope.Memory
|
|
|
|
stack := scope.Stack
|
|
|
|
|
2021-05-06 17:37:38 +00:00
|
|
|
select {
|
|
|
|
case <-l.ctx.Done():
|
2021-12-15 13:19:58 +00:00
|
|
|
return
|
2020-08-19 11:46:20 +00:00
|
|
|
default:
|
|
|
|
}
|
2021-05-06 17:37:38 +00:00
|
|
|
// check if already accumulated the specified number of logs
|
|
|
|
if l.cfg.Limit != 0 && l.cfg.Limit <= len(l.logs) {
|
2021-12-15 13:19:58 +00:00
|
|
|
return
|
2021-05-06 17:37:38 +00:00
|
|
|
}
|
|
|
|
if !l.firstCapture {
|
|
|
|
l.stream.WriteMore()
|
|
|
|
} else {
|
|
|
|
l.firstCapture = false
|
|
|
|
}
|
2021-07-23 21:37:54 +00:00
|
|
|
var outputStorage bool
|
2021-05-06 17:37:38 +00:00
|
|
|
if !l.cfg.DisableStorage {
|
|
|
|
// initialise new changed values storage container for this contract
|
|
|
|
// if not present.
|
|
|
|
if l.storage[contract.Address()] == nil {
|
|
|
|
l.storage[contract.Address()] = make(vm.Storage)
|
|
|
|
}
|
|
|
|
// capture SLOAD opcodes and record the read entry in the local storage
|
|
|
|
if op == vm.SLOAD && stack.Len() >= 1 {
|
|
|
|
var (
|
|
|
|
address = common.Hash(stack.Data[stack.Len()-1].Bytes32())
|
|
|
|
value uint256.Int
|
|
|
|
)
|
2021-12-06 14:58:53 +00:00
|
|
|
env.IntraBlockState().GetState(contract.Address(), &address, &value)
|
2022-08-13 11:51:25 +00:00
|
|
|
l.storage[contract.Address()][address] = value.Bytes32()
|
2021-07-23 21:37:54 +00:00
|
|
|
outputStorage = true
|
2021-05-06 17:37:38 +00:00
|
|
|
}
|
|
|
|
// capture SSTORE opcodes and record the written entry in the local storage.
|
|
|
|
if op == vm.SSTORE && stack.Len() >= 2 {
|
|
|
|
var (
|
|
|
|
value = common.Hash(stack.Data[stack.Len()-2].Bytes32())
|
|
|
|
address = common.Hash(stack.Data[stack.Len()-1].Bytes32())
|
|
|
|
)
|
|
|
|
l.storage[contract.Address()][address] = value
|
2021-07-23 21:37:54 +00:00
|
|
|
outputStorage = true
|
2021-05-06 17:37:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
// create a new snapshot of the EVM.
|
|
|
|
l.stream.WriteObjectStart()
|
|
|
|
l.stream.WriteObjectField("pc")
|
|
|
|
l.stream.WriteUint64(pc)
|
|
|
|
l.stream.WriteMore()
|
|
|
|
l.stream.WriteObjectField("op")
|
|
|
|
l.stream.WriteString(op.String())
|
|
|
|
l.stream.WriteMore()
|
|
|
|
l.stream.WriteObjectField("gas")
|
|
|
|
l.stream.WriteUint64(gas)
|
|
|
|
l.stream.WriteMore()
|
|
|
|
l.stream.WriteObjectField("gasCost")
|
|
|
|
l.stream.WriteUint64(cost)
|
|
|
|
l.stream.WriteMore()
|
|
|
|
l.stream.WriteObjectField("depth")
|
|
|
|
l.stream.WriteInt(depth)
|
|
|
|
if err != nil {
|
|
|
|
l.stream.WriteMore()
|
|
|
|
l.stream.WriteObjectField("error")
|
2021-08-08 12:28:03 +00:00
|
|
|
l.stream.WriteObjectStart()
|
|
|
|
l.stream.WriteObjectEnd()
|
|
|
|
//l.stream.WriteString(err.Error())
|
2021-05-06 17:37:38 +00:00
|
|
|
}
|
|
|
|
if !l.cfg.DisableStack {
|
|
|
|
l.stream.WriteMore()
|
|
|
|
l.stream.WriteObjectField("stack")
|
|
|
|
l.stream.WriteArrayStart()
|
|
|
|
for i, stackValue := range stack.Data {
|
|
|
|
if i > 0 {
|
|
|
|
l.stream.WriteMore()
|
|
|
|
}
|
2021-07-23 21:37:54 +00:00
|
|
|
l.stream.WriteString(stackValue.String())
|
2021-05-06 17:37:38 +00:00
|
|
|
}
|
|
|
|
l.stream.WriteArrayEnd()
|
|
|
|
}
|
|
|
|
if !l.cfg.DisableMemory {
|
|
|
|
memData := memory.Data()
|
|
|
|
l.stream.WriteMore()
|
|
|
|
l.stream.WriteObjectField("memory")
|
|
|
|
l.stream.WriteArrayStart()
|
|
|
|
for i := 0; i+32 <= len(memData); i += 32 {
|
|
|
|
if i > 0 {
|
|
|
|
l.stream.WriteMore()
|
|
|
|
}
|
2021-08-17 19:50:52 +00:00
|
|
|
l.stream.WriteString(string(l.hexEncodeBuf[0:hex.Encode(l.hexEncodeBuf[:], memData[i:i+32])]))
|
2021-05-06 17:37:38 +00:00
|
|
|
}
|
|
|
|
l.stream.WriteArrayEnd()
|
|
|
|
}
|
2021-07-23 21:37:54 +00:00
|
|
|
if outputStorage {
|
2021-05-06 17:37:38 +00:00
|
|
|
l.stream.WriteMore()
|
|
|
|
l.stream.WriteObjectField("storage")
|
|
|
|
l.stream.WriteObjectStart()
|
|
|
|
first := true
|
2021-07-03 13:34:23 +00:00
|
|
|
// Sort storage by locations for easier comparison with geth
|
|
|
|
if l.locations != nil {
|
|
|
|
l.locations = l.locations[:0]
|
|
|
|
}
|
|
|
|
s := l.storage[contract.Address()]
|
|
|
|
for loc := range s {
|
|
|
|
l.locations = append(l.locations, loc)
|
|
|
|
}
|
|
|
|
sort.Sort(l.locations)
|
|
|
|
for _, loc := range l.locations {
|
|
|
|
value := s[loc]
|
2021-05-06 17:37:38 +00:00
|
|
|
if first {
|
|
|
|
first = false
|
|
|
|
} else {
|
|
|
|
l.stream.WriteMore()
|
|
|
|
}
|
2021-08-17 19:50:52 +00:00
|
|
|
l.stream.WriteObjectField(string(l.hexEncodeBuf[0:hex.Encode(l.hexEncodeBuf[:], loc[:])]))
|
|
|
|
l.stream.WriteString(string(l.hexEncodeBuf[0:hex.Encode(l.hexEncodeBuf[:], value[:])]))
|
2021-05-06 17:37:38 +00:00
|
|
|
}
|
|
|
|
l.stream.WriteObjectEnd()
|
|
|
|
}
|
|
|
|
l.stream.WriteObjectEnd()
|
2021-12-15 13:19:58 +00:00
|
|
|
_ = l.stream.Flush()
|
2021-05-06 17:37:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// CaptureFault implements the Tracer interface to trace an execution fault
|
|
|
|
// while running an opcode.
|
2021-12-15 13:19:58 +00:00
|
|
|
func (l *JsonStreamLogger) CaptureFault(env *vm.EVM, pc uint64, op vm.OpCode, gas, cost uint64, scope *vm.ScopeContext, depth int, err error) {
|
2021-05-06 17:37:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// CaptureEnd is called after the call finishes to finalize the tracing.
|
2021-12-15 13:19:58 +00:00
|
|
|
func (l *JsonStreamLogger) CaptureEnd(depth int, output []byte, startGas, endGas uint64, t time.Duration, err error) {
|
2021-05-06 17:37:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (l *JsonStreamLogger) CaptureSelfDestruct(from common.Address, to common.Address, value *big.Int) {
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l *JsonStreamLogger) CaptureAccountRead(account common.Address) error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l *JsonStreamLogger) CaptureAccountWrite(account common.Address) error {
|
|
|
|
return nil
|
2020-08-19 11:46:20 +00:00
|
|
|
}
|