mirror of https://github.com/status-im/op-geth.git
core: differentiate receipt concensus and storage decoding
This commit is contained in:
parent
b99fe27f8b
commit
42c8afd440
|
@ -111,7 +111,7 @@ func (self *BlockProcessor) ApplyTransaction(gp GasPool, statedb *state.StateDB,
|
||||||
}
|
}
|
||||||
|
|
||||||
logs := statedb.GetLogs(tx.Hash())
|
logs := statedb.GetLogs(tx.Hash())
|
||||||
receipt.SetLogs(logs)
|
receipt.Logs = logs
|
||||||
receipt.Bloom = types.CreateBloom(types.Receipts{receipt})
|
receipt.Bloom = types.CreateBloom(types.Receipts{receipt})
|
||||||
|
|
||||||
glog.V(logger.Debug).Infoln(receipt)
|
glog.V(logger.Debug).Infoln(receipt)
|
||||||
|
@ -364,7 +364,7 @@ func (sm *BlockProcessor) GetLogs(block *types.Block) (logs vm.Logs, err error)
|
||||||
receipts := GetBlockReceipts(sm.chainDb, block.Hash())
|
receipts := GetBlockReceipts(sm.chainDb, block.Hash())
|
||||||
// coalesce logs
|
// coalesce logs
|
||||||
for _, receipt := range receipts {
|
for _, receipt := range receipts {
|
||||||
logs = append(logs, receipt.Logs()...)
|
logs = append(logs, receipt.Logs...)
|
||||||
}
|
}
|
||||||
return logs, nil
|
return logs, nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -70,7 +70,7 @@ func TestPutReceipt(t *testing.T) {
|
||||||
hash[0] = 2
|
hash[0] = 2
|
||||||
|
|
||||||
receipt := new(types.Receipt)
|
receipt := new(types.Receipt)
|
||||||
receipt.SetLogs(vm.Logs{&vm.Log{
|
receipt.Logs = vm.Logs{&vm.Log{
|
||||||
Address: addr,
|
Address: addr,
|
||||||
Topics: []common.Hash{hash},
|
Topics: []common.Hash{hash},
|
||||||
Data: []byte("hi"),
|
Data: []byte("hi"),
|
||||||
|
@ -79,7 +79,7 @@ func TestPutReceipt(t *testing.T) {
|
||||||
TxIndex: 0,
|
TxIndex: 0,
|
||||||
BlockHash: hash,
|
BlockHash: hash,
|
||||||
Index: 0,
|
Index: 0,
|
||||||
}})
|
}}
|
||||||
|
|
||||||
PutReceipts(db, types.Receipts{receipt})
|
PutReceipts(db, types.Receipts{receipt})
|
||||||
receipt = GetReceipt(db, common.Hash{})
|
receipt = GetReceipt(db, common.Hash{})
|
||||||
|
|
|
@ -99,7 +99,7 @@ func (b *BlockGen) AddTx(tx *types.Transaction) {
|
||||||
b.header.GasUsed.Add(b.header.GasUsed, gas)
|
b.header.GasUsed.Add(b.header.GasUsed, gas)
|
||||||
receipt := types.NewReceipt(root.Bytes(), b.header.GasUsed)
|
receipt := types.NewReceipt(root.Bytes(), b.header.GasUsed)
|
||||||
logs := b.statedb.GetLogs(tx.Hash())
|
logs := b.statedb.GetLogs(tx.Hash())
|
||||||
receipt.SetLogs(logs)
|
receipt.Logs = logs
|
||||||
receipt.Bloom = types.CreateBloom(types.Receipts{receipt})
|
receipt.Bloom = types.CreateBloom(types.Receipts{receipt})
|
||||||
b.txs = append(b.txs, tx)
|
b.txs = append(b.txs, tx)
|
||||||
b.receipts = append(b.receipts, receipt)
|
b.receipts = append(b.receipts, receipt)
|
||||||
|
|
|
@ -140,12 +140,16 @@ func GetBlockReceipts(db ethdb.Database, hash common.Hash) types.Receipts {
|
||||||
if len(data) == 0 {
|
if len(data) == 0 {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
receipts := new(types.Receipts)
|
rs := []*types.ReceiptForStorage{}
|
||||||
if err := rlp.DecodeBytes(data, receipts); err != nil {
|
if err := rlp.DecodeBytes(data, &rs); err != nil {
|
||||||
glog.V(logger.Error).Infof("invalid receipt array RLP for hash %x: %v", hash, err)
|
glog.V(logger.Error).Infof("invalid receipt array RLP for hash %x: %v", hash, err)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
return *receipts
|
receipts := make(types.Receipts, len(rs))
|
||||||
|
for i, receipt := range rs {
|
||||||
|
receipts[i] = (*types.Receipt)(receipt)
|
||||||
|
}
|
||||||
|
return receipts
|
||||||
}
|
}
|
||||||
|
|
||||||
// PutBlockReceipts stores the block's transactions associated receipts
|
// PutBlockReceipts stores the block's transactions associated receipts
|
||||||
|
|
|
@ -72,7 +72,7 @@ func (b Bloom) TestBytes(test []byte) bool {
|
||||||
func CreateBloom(receipts Receipts) Bloom {
|
func CreateBloom(receipts Receipts) Bloom {
|
||||||
bin := new(big.Int)
|
bin := new(big.Int)
|
||||||
for _, receipt := range receipts {
|
for _, receipt := range receipts {
|
||||||
bin.Or(bin, LogsBloom(receipt.logs))
|
bin.Or(bin, LogsBloom(receipt.Logs))
|
||||||
}
|
}
|
||||||
|
|
||||||
return BytesToBloom(bin.Bytes())
|
return BytesToBloom(bin.Bytes())
|
||||||
|
|
|
@ -17,7 +17,6 @@
|
||||||
package types
|
package types
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"math/big"
|
"math/big"
|
||||||
|
@ -27,89 +26,116 @@ import (
|
||||||
"github.com/ethereum/go-ethereum/rlp"
|
"github.com/ethereum/go-ethereum/rlp"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// Receipt represents the results of a transaction.
|
||||||
type Receipt struct {
|
type Receipt struct {
|
||||||
|
// Consensus fields
|
||||||
PostState []byte
|
PostState []byte
|
||||||
CumulativeGasUsed *big.Int
|
CumulativeGasUsed *big.Int
|
||||||
Bloom Bloom
|
Bloom Bloom
|
||||||
TxHash common.Hash
|
Logs vm.Logs
|
||||||
ContractAddress common.Address
|
|
||||||
logs vm.Logs
|
// Implementation fields
|
||||||
GasUsed *big.Int
|
TxHash common.Hash
|
||||||
|
ContractAddress common.Address
|
||||||
|
GasUsed *big.Int
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// NewReceipt creates a barebone transaction receipt, copying the init fields.
|
||||||
func NewReceipt(root []byte, cumalativeGasUsed *big.Int) *Receipt {
|
func NewReceipt(root []byte, cumalativeGasUsed *big.Int) *Receipt {
|
||||||
return &Receipt{PostState: common.CopyBytes(root), CumulativeGasUsed: new(big.Int).Set(cumalativeGasUsed)}
|
return &Receipt{PostState: common.CopyBytes(root), CumulativeGasUsed: new(big.Int).Set(cumalativeGasUsed)}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (self *Receipt) SetLogs(logs vm.Logs) {
|
// EncodeRLP implements rlp.Encoder, and flattens the consensus fields of a receipt
|
||||||
self.logs = logs
|
// into an RLP stream.
|
||||||
|
func (r *Receipt) EncodeRLP(w io.Writer) error {
|
||||||
|
return rlp.Encode(w, []interface{}{r.PostState, r.CumulativeGasUsed, r.Bloom, r.Logs})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (self *Receipt) Logs() vm.Logs {
|
// DecodeRLP implements rlp.Decoder, and loads the consensus fields of a receipt
|
||||||
return self.logs
|
// from an RLP stream.
|
||||||
|
func (r *Receipt) DecodeRLP(s *rlp.Stream) error {
|
||||||
|
var receipt struct {
|
||||||
|
PostState []byte
|
||||||
|
CumulativeGasUsed *big.Int
|
||||||
|
Bloom Bloom
|
||||||
|
Logs vm.Logs
|
||||||
|
}
|
||||||
|
if err := s.Decode(&receipt); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
r.PostState, r.CumulativeGasUsed, r.Bloom, r.Logs = receipt.PostState, receipt.CumulativeGasUsed, receipt.Bloom, receipt.Logs
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (self *Receipt) EncodeRLP(w io.Writer) error {
|
// RlpEncode implements common.RlpEncode required for SHA derivation.
|
||||||
return rlp.Encode(w, []interface{}{self.PostState, self.CumulativeGasUsed, self.Bloom, self.logs})
|
func (r *Receipt) RlpEncode() []byte {
|
||||||
|
bytes, err := rlp.EncodeToBytes(r)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
return bytes
|
||||||
}
|
}
|
||||||
|
|
||||||
func (self *Receipt) DecodeRLP(s *rlp.Stream) error {
|
// String implements the Stringer interface.
|
||||||
var r struct {
|
func (r *Receipt) String() string {
|
||||||
|
return fmt.Sprintf("receipt{med=%x cgas=%v bloom=%x logs=%v}", r.PostState, r.CumulativeGasUsed, r.Bloom, r.Logs)
|
||||||
|
}
|
||||||
|
|
||||||
|
// ReceiptForStorage is a wrapper around a Receipt that flattens and parses the
|
||||||
|
// entire content of a receipt, opposed to only the consensus fields originally.
|
||||||
|
type ReceiptForStorage Receipt
|
||||||
|
|
||||||
|
// EncodeRLP implements rlp.Encoder, and flattens all content fields of a receipt
|
||||||
|
// into an RLP stream.
|
||||||
|
func (r *ReceiptForStorage) EncodeRLP(w io.Writer) error {
|
||||||
|
logs := make([]*vm.LogForStorage, len(r.Logs))
|
||||||
|
for i, log := range r.Logs {
|
||||||
|
logs[i] = (*vm.LogForStorage)(log)
|
||||||
|
}
|
||||||
|
return rlp.Encode(w, []interface{}{r.PostState, r.CumulativeGasUsed, r.Bloom, r.TxHash, r.ContractAddress, logs, r.GasUsed})
|
||||||
|
}
|
||||||
|
|
||||||
|
// DecodeRLP implements rlp.Decoder, and loads the consensus fields of a receipt
|
||||||
|
// from an RLP stream.
|
||||||
|
func (r *ReceiptForStorage) DecodeRLP(s *rlp.Stream) error {
|
||||||
|
var receipt struct {
|
||||||
PostState []byte
|
PostState []byte
|
||||||
CumulativeGasUsed *big.Int
|
CumulativeGasUsed *big.Int
|
||||||
Bloom Bloom
|
Bloom Bloom
|
||||||
TxHash common.Hash
|
TxHash common.Hash
|
||||||
ContractAddress common.Address
|
ContractAddress common.Address
|
||||||
Logs vm.Logs
|
Logs []*vm.LogForStorage
|
||||||
GasUsed *big.Int
|
GasUsed *big.Int
|
||||||
}
|
}
|
||||||
if err := s.Decode(&r); err != nil {
|
if err := s.Decode(&receipt); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
self.PostState, self.CumulativeGasUsed, self.Bloom, self.TxHash, self.ContractAddress, self.logs, self.GasUsed = r.PostState, r.CumulativeGasUsed, r.Bloom, r.TxHash, r.ContractAddress, r.Logs, r.GasUsed
|
// Assign the consensus fields
|
||||||
|
r.PostState, r.CumulativeGasUsed, r.Bloom = receipt.PostState, receipt.CumulativeGasUsed, receipt.Bloom
|
||||||
|
r.Logs = make(vm.Logs, len(receipt.Logs))
|
||||||
|
for i, log := range receipt.Logs {
|
||||||
|
r.Logs[i] = (*vm.Log)(log)
|
||||||
|
}
|
||||||
|
// Assign the implementation fields
|
||||||
|
r.TxHash, r.ContractAddress, r.GasUsed = receipt.TxHash, receipt.ContractAddress, receipt.GasUsed
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
type ReceiptForStorage Receipt
|
// Receipts is a wrapper around a Receipt array to implement types.DerivableList.
|
||||||
|
|
||||||
func (self *ReceiptForStorage) EncodeRLP(w io.Writer) error {
|
|
||||||
storageLogs := make([]*vm.LogForStorage, len(self.logs))
|
|
||||||
for i, log := range self.logs {
|
|
||||||
storageLogs[i] = (*vm.LogForStorage)(log)
|
|
||||||
}
|
|
||||||
return rlp.Encode(w, []interface{}{self.PostState, self.CumulativeGasUsed, self.Bloom, self.TxHash, self.ContractAddress, storageLogs, self.GasUsed})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (self *Receipt) RlpEncode() []byte {
|
|
||||||
bytes, err := rlp.EncodeToBytes(self)
|
|
||||||
if err != nil {
|
|
||||||
fmt.Println("TMP -- RECEIPT ENCODE ERROR", err)
|
|
||||||
}
|
|
||||||
return bytes
|
|
||||||
}
|
|
||||||
|
|
||||||
func (self *Receipt) Cmp(other *Receipt) bool {
|
|
||||||
if bytes.Compare(self.PostState, other.PostState) != 0 {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
|
|
||||||
func (self *Receipt) String() string {
|
|
||||||
return fmt.Sprintf("receipt{med=%x cgas=%v bloom=%x logs=%v}", self.PostState, self.CumulativeGasUsed, self.Bloom, self.logs)
|
|
||||||
}
|
|
||||||
|
|
||||||
type Receipts []*Receipt
|
type Receipts []*Receipt
|
||||||
|
|
||||||
func (self Receipts) RlpEncode() []byte {
|
// RlpEncode implements common.RlpEncode required for SHA derivation.
|
||||||
bytes, err := rlp.EncodeToBytes(self)
|
func (r Receipts) RlpEncode() []byte {
|
||||||
|
bytes, err := rlp.EncodeToBytes(r)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Println("TMP -- RECEIPTS ENCODE ERROR", err)
|
panic(err)
|
||||||
}
|
}
|
||||||
return bytes
|
return bytes
|
||||||
}
|
}
|
||||||
|
|
||||||
func (self Receipts) Len() int { return len(self) }
|
// Len returns the number of receipts in this list.
|
||||||
func (self Receipts) GetRlp(i int) []byte { return common.Rlp(self[i]) }
|
func (r Receipts) Len() int { return len(r) }
|
||||||
|
|
||||||
|
// GetRlp returns the RLP encoding of one receipt from the list.
|
||||||
|
func (r Receipts) GetRlp(i int) []byte { return common.Rlp(r[i]) }
|
||||||
|
|
|
@ -40,27 +40,30 @@ func NewLog(address common.Address, topics []common.Hash, data []byte, number ui
|
||||||
return &Log{Address: address, Topics: topics, Data: data, Number: number}
|
return &Log{Address: address, Topics: topics, Data: data, Number: number}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (self *Log) EncodeRLP(w io.Writer) error {
|
func (l *Log) EncodeRLP(w io.Writer) error {
|
||||||
return rlp.Encode(w, []interface{}{self.Address, self.Topics, self.Data})
|
return rlp.Encode(w, []interface{}{l.Address, l.Topics, l.Data})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (self *Log) String() string {
|
func (l *Log) DecodeRLP(s *rlp.Stream) error {
|
||||||
return fmt.Sprintf(`log: %x %x %x %x %d %x %d`, self.Address, self.Topics, self.Data, self.TxHash, self.TxIndex, self.BlockHash, self.Index)
|
var log struct {
|
||||||
|
Address common.Address
|
||||||
|
Topics []common.Hash
|
||||||
|
Data []byte
|
||||||
|
}
|
||||||
|
if err := s.Decode(&log); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
l.Address, l.Topics, l.Data = log.Address, log.Topics, log.Data
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l *Log) String() string {
|
||||||
|
return fmt.Sprintf(`log: %x %x %x %x %d %x %d`, l.Address, l.Topics, l.Data, l.TxHash, l.TxIndex, l.BlockHash, l.Index)
|
||||||
}
|
}
|
||||||
|
|
||||||
type Logs []*Log
|
type Logs []*Log
|
||||||
|
|
||||||
|
// LogForStorage is a wrapper around a Log that flattens and parses the entire
|
||||||
|
// content of a log, opposed to only the consensus fields originally (by hiding
|
||||||
|
// the rlp interface methods).
|
||||||
type LogForStorage Log
|
type LogForStorage Log
|
||||||
|
|
||||||
func (self *LogForStorage) EncodeRLP(w io.Writer) error {
|
|
||||||
return rlp.Encode(w, []interface{}{
|
|
||||||
self.Address,
|
|
||||||
self.Topics,
|
|
||||||
self.Data,
|
|
||||||
self.Number,
|
|
||||||
self.TxHash,
|
|
||||||
self.TxIndex,
|
|
||||||
self.BlockHash,
|
|
||||||
self.Index,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
|
@ -138,7 +138,7 @@ func (self *Filter) getLogs(start, end uint64) (logs vm.Logs) {
|
||||||
unfiltered vm.Logs
|
unfiltered vm.Logs
|
||||||
)
|
)
|
||||||
for _, receipt := range receipts {
|
for _, receipt := range receipts {
|
||||||
unfiltered = append(unfiltered, receipt.Logs()...)
|
unfiltered = append(unfiltered, receipt.Logs...)
|
||||||
}
|
}
|
||||||
logs = append(logs, self.FilterLogs(unfiltered)...)
|
logs = append(logs, self.FilterLogs(unfiltered)...)
|
||||||
}
|
}
|
||||||
|
|
|
@ -453,8 +453,8 @@ func NewReceiptRes(rec *types.Receipt) *ReceiptRes {
|
||||||
v.ContractAddress = newHexData(rec.ContractAddress)
|
v.ContractAddress = newHexData(rec.ContractAddress)
|
||||||
}
|
}
|
||||||
|
|
||||||
logs := make([]interface{}, len(rec.Logs()))
|
logs := make([]interface{}, len(rec.Logs))
|
||||||
for i, log := range rec.Logs() {
|
for i, log := range rec.Logs {
|
||||||
logs[i] = NewLogRes(log)
|
logs[i] = NewLogRes(log)
|
||||||
}
|
}
|
||||||
v.Logs = &logs
|
v.Logs = &logs
|
||||||
|
|
Loading…
Reference in New Issue