You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
174 lines
5.1 KiB
174 lines
5.1 KiB
3 years ago
|
// Copyright 2017 The go-ethereum Authors
|
||
|
// This file is part of the go-ethereum library.
|
||
|
//
|
||
|
// The go-ethereum library is free software: you can redistribute it and/or modify
|
||
|
// it under the terms of the GNU Lesser General Public License as published by
|
||
|
// the Free Software Foundation, either version 3 of the License, or
|
||
|
// (at your option) any later version.
|
||
|
//
|
||
|
// The go-ethereum library is distributed in the hope that it will be useful,
|
||
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
|
// GNU Lesser General Public License for more details.
|
||
|
//
|
||
|
// You should have received a copy of the GNU Lesser General Public License
|
||
|
// along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
|
||
|
|
||
|
package tracers
|
||
|
|
||
|
import (
|
||
|
"encoding/json"
|
||
|
"fmt"
|
||
|
"math/big"
|
||
|
|
||
|
"github.com/ethereum/go-ethereum/common"
|
||
|
"github.com/ethereum/go-ethereum/rlp"
|
||
|
"github.com/harmony-one/harmony/crypto/hash"
|
||
|
)
|
||
|
|
||
|
type ActionStorage struct {
|
||
|
Subtraces uint
|
||
|
TraceAddress []uint
|
||
|
TraceData []byte
|
||
|
}
|
||
|
|
||
|
func (storage *ActionStorage) appendByte(byt byte) {
|
||
|
storage.TraceData = append(storage.TraceData, byt)
|
||
|
}
|
||
|
|
||
|
func (storage *ActionStorage) appendFixed(data []byte) {
|
||
|
storage.TraceData = append(storage.TraceData, data...)
|
||
|
}
|
||
|
func (storage *ActionStorage) appendNumber(num *big.Int) {
|
||
|
bytes, _ := rlp.EncodeToBytes(num)
|
||
|
storage.appendByte(uint8(len(bytes)))
|
||
|
storage.appendFixed(bytes)
|
||
|
}
|
||
|
|
||
|
func (storage *ActionStorage) readByte() byte {
|
||
|
val := storage.TraceData[0]
|
||
|
storage.TraceData = storage.TraceData[1:]
|
||
|
return val
|
||
|
}
|
||
|
func (storage *ActionStorage) readFixedData(size uint) []byte {
|
||
|
fixedData := storage.TraceData[:size]
|
||
|
storage.TraceData = storage.TraceData[size:]
|
||
|
return fixedData
|
||
|
}
|
||
|
func (storage *ActionStorage) readNumber() *big.Int {
|
||
|
size := storage.readByte()
|
||
|
bytes := storage.readFixedData(uint(size))
|
||
|
var num big.Int
|
||
|
rlp.DecodeBytes(bytes, &num)
|
||
|
return &num
|
||
|
}
|
||
|
|
||
|
type TxStorage struct {
|
||
|
Hash common.Hash
|
||
|
Storages []*ActionStorage
|
||
|
}
|
||
|
|
||
|
type TraceBlockStorage struct {
|
||
|
Hash common.Hash
|
||
|
Number uint64
|
||
|
AddressTable []common.Address // address table
|
||
|
DataKeyTable []common.Hash // data key table
|
||
|
dataValueTable [][]byte // data, store in db, avoid RLPEncode
|
||
|
TraceStorages [][]byte // trace data, lenth equal the number of transaction in a block
|
||
|
addressIndex map[common.Address]int // address index in AddressTable
|
||
|
dataIndex map[common.Hash]int // data index in DataKeyTable
|
||
|
}
|
||
|
|
||
|
func (ts *TraceBlockStorage) getData(i int) []byte {
|
||
|
return ts.dataValueTable[i]
|
||
|
}
|
||
|
|
||
|
func (ts *TraceBlockStorage) getAddress(i int) common.Address {
|
||
|
return ts.AddressTable[i]
|
||
|
}
|
||
|
|
||
|
func (ts *TraceBlockStorage) indexData(data []byte) int {
|
||
|
key := hash.Keccak256Hash(data)
|
||
|
if index, exist := ts.dataIndex[key]; exist {
|
||
|
return index
|
||
|
}
|
||
|
index := len(ts.DataKeyTable)
|
||
|
ts.DataKeyTable = append(ts.DataKeyTable, key)
|
||
|
ts.dataValueTable = append(ts.dataValueTable, data)
|
||
|
ts.dataIndex[key] = index
|
||
|
return index
|
||
|
}
|
||
|
|
||
|
func (ts *TraceBlockStorage) indexAddress(address common.Address) int {
|
||
|
if index, exist := ts.addressIndex[address]; exist {
|
||
|
return index
|
||
|
}
|
||
|
index := len(ts.addressIndex)
|
||
|
ts.AddressTable = append(ts.AddressTable, address)
|
||
|
ts.addressIndex[address] = index
|
||
|
return index
|
||
|
}
|
||
|
|
||
|
func (ts *TraceBlockStorage) toDB(write func([]byte, []byte)) {
|
||
|
for index, key := range ts.DataKeyTable {
|
||
|
write(key[:], ts.dataValueTable[index])
|
||
|
}
|
||
|
bytes, _ := rlp.EncodeToBytes(ts)
|
||
|
write(ts.Hash[:], bytes)
|
||
|
}
|
||
|
|
||
|
func (ts *TraceBlockStorage) fromDB(read func([]byte) []byte, hash common.Hash) {
|
||
|
bytes := read(hash[:])
|
||
|
rlp.DecodeBytes(bytes, ts)
|
||
|
for _, key := range ts.DataKeyTable {
|
||
|
ts.dataValueTable = append(ts.dataValueTable, read(key[:]))
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func (ts *TraceBlockStorage) txJson(index int) []json.RawMessage {
|
||
|
var results []json.RawMessage
|
||
|
var txStorage TxStorage
|
||
|
b := ts.TraceStorages[index]
|
||
|
rlp.DecodeBytes(b, &txStorage)
|
||
|
|
||
|
headPiece := fmt.Sprintf(
|
||
|
`"blockNumber":%d,"blockHash":"%s","transactionHash":"%s","transactionPosition":%d`,
|
||
|
ts.Number, ts.Hash.Hex(), txStorage.Hash.Hex(), index,
|
||
|
)
|
||
|
|
||
|
for _, acStorage := range txStorage.Storages {
|
||
|
ac := &action{}
|
||
|
ac.fromStorage(ts, acStorage)
|
||
|
|
||
|
typStr, acStr, outStr := ac.toJsonStr()
|
||
|
if acStr == nil {
|
||
|
//err = errors.New("tracer internal failure")
|
||
|
return nil
|
||
|
}
|
||
|
traceStr, _ := json.Marshal(acStorage.TraceAddress)
|
||
|
bodyPiece := fmt.Sprintf(
|
||
|
`,"subtraces":%d,"traceAddress":%s,"type":"%s","action":%s`,
|
||
|
acStorage.Subtraces, string(traceStr), typStr, *acStr,
|
||
|
)
|
||
|
var resultPiece string
|
||
|
if ac.err != nil {
|
||
|
resultPiece = fmt.Sprintf(`,"error":"Reverted","revert":"0x%x"`, ac.revert)
|
||
|
} else if outStr != nil {
|
||
|
resultPiece = fmt.Sprintf(`,"result":%s`, *outStr)
|
||
|
} else {
|
||
|
resultPiece = `,"result":null`
|
||
|
}
|
||
|
jstr := "{" + headPiece + bodyPiece + resultPiece + "}"
|
||
|
results = append(results, json.RawMessage(jstr))
|
||
|
}
|
||
|
return results
|
||
|
}
|
||
|
|
||
|
func (ts *TraceBlockStorage) toJson() []json.RawMessage {
|
||
|
var results []json.RawMessage
|
||
|
for i := range ts.TraceStorages {
|
||
|
results = append(results, ts.txJson(i)...)
|
||
|
}
|
||
|
return results
|
||
|
}
|