forked from cerc-io/plugeth
core/types: add block location fields to receipt (#17662)
Solves #15210 without changing consensus, in a backwards compatible way, by adding tx inclusion information to the Receipt struct.
This commit is contained in:
parent
42e2c586fd
commit
7fb89697fd
@ -810,6 +810,11 @@ func SetReceiptsData(config *params.ChainConfig, block *types.Block, receipts ty
|
|||||||
// The transaction hash can be retrieved from the transaction itself
|
// The transaction hash can be retrieved from the transaction itself
|
||||||
receipts[j].TxHash = transactions[j].Hash()
|
receipts[j].TxHash = transactions[j].Hash()
|
||||||
|
|
||||||
|
// block location fields
|
||||||
|
receipts[j].BlockHash = block.Hash()
|
||||||
|
receipts[j].BlockNumber = block.Number()
|
||||||
|
receipts[j].TransactionIndex = uint(j)
|
||||||
|
|
||||||
// The contract address can be derived from the transaction itself
|
// The contract address can be derived from the transaction itself
|
||||||
if transactions[j].To() == nil {
|
if transactions[j].To() == nil {
|
||||||
// Deriving the signer is expensive, only do if it's actually needed
|
// Deriving the signer is expensive, only do if it's actually needed
|
||||||
|
@ -325,6 +325,9 @@ func ReadReceipts(db ethdb.Reader, hash common.Hash, number uint64) types.Receip
|
|||||||
logIndex += 1
|
logIndex += 1
|
||||||
}
|
}
|
||||||
receipts[i] = (*types.Receipt)(receipt)
|
receipts[i] = (*types.Receipt)(receipt)
|
||||||
|
receipts[i].BlockHash = hash
|
||||||
|
receipts[i].BlockNumber = big.NewInt(0).SetUint64(number)
|
||||||
|
receipts[i].TransactionIndex = uint(i)
|
||||||
}
|
}
|
||||||
return receipts
|
return receipts
|
||||||
}
|
}
|
||||||
|
@ -237,6 +237,16 @@ func (self *StateDB) GetNonce(addr common.Address) uint64 {
|
|||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TxIndex returns the current transaction index set by Prepare.
|
||||||
|
func (self *StateDB) TxIndex() int {
|
||||||
|
return self.txIndex
|
||||||
|
}
|
||||||
|
|
||||||
|
// BlockHash returns the current block hash set by Prepare.
|
||||||
|
func (self *StateDB) BlockHash() common.Hash {
|
||||||
|
return self.bhash
|
||||||
|
}
|
||||||
|
|
||||||
func (self *StateDB) GetCode(addr common.Address) []byte {
|
func (self *StateDB) GetCode(addr common.Address) []byte {
|
||||||
stateObject := self.getStateObject(addr)
|
stateObject := self.getStateObject(addr)
|
||||||
if stateObject != nil {
|
if stateObject != nil {
|
||||||
|
@ -121,6 +121,9 @@ func ApplyTransaction(config *params.ChainConfig, bc ChainContext, author *commo
|
|||||||
// Set the receipt logs and create a bloom for filtering
|
// Set the receipt logs and create a bloom for filtering
|
||||||
receipt.Logs = statedb.GetLogs(tx.Hash())
|
receipt.Logs = statedb.GetLogs(tx.Hash())
|
||||||
receipt.Bloom = types.CreateBloom(types.Receipts{receipt})
|
receipt.Bloom = types.CreateBloom(types.Receipts{receipt})
|
||||||
|
receipt.BlockHash = statedb.BlockHash()
|
||||||
|
receipt.BlockNumber = header.Number
|
||||||
|
receipt.TransactionIndex = uint(statedb.TxIndex())
|
||||||
|
|
||||||
return receipt, gas, err
|
return receipt, gas, err
|
||||||
}
|
}
|
||||||
|
@ -5,6 +5,7 @@ package types
|
|||||||
import (
|
import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"errors"
|
"errors"
|
||||||
|
"math/big"
|
||||||
|
|
||||||
"github.com/ethereum/go-ethereum/common"
|
"github.com/ethereum/go-ethereum/common"
|
||||||
"github.com/ethereum/go-ethereum/common/hexutil"
|
"github.com/ethereum/go-ethereum/common/hexutil"
|
||||||
@ -23,6 +24,9 @@ func (r Receipt) MarshalJSON() ([]byte, error) {
|
|||||||
TxHash common.Hash `json:"transactionHash" gencodec:"required"`
|
TxHash common.Hash `json:"transactionHash" gencodec:"required"`
|
||||||
ContractAddress common.Address `json:"contractAddress"`
|
ContractAddress common.Address `json:"contractAddress"`
|
||||||
GasUsed hexutil.Uint64 `json:"gasUsed" gencodec:"required"`
|
GasUsed hexutil.Uint64 `json:"gasUsed" gencodec:"required"`
|
||||||
|
BlockHash common.Hash `json:"blockHash,omitempty"`
|
||||||
|
BlockNumber *hexutil.Big `json:"blockNumber,omitempty"`
|
||||||
|
TransactionIndex hexutil.Uint `json:"transactionIndex"`
|
||||||
}
|
}
|
||||||
var enc Receipt
|
var enc Receipt
|
||||||
enc.PostState = r.PostState
|
enc.PostState = r.PostState
|
||||||
@ -33,6 +37,9 @@ func (r Receipt) MarshalJSON() ([]byte, error) {
|
|||||||
enc.TxHash = r.TxHash
|
enc.TxHash = r.TxHash
|
||||||
enc.ContractAddress = r.ContractAddress
|
enc.ContractAddress = r.ContractAddress
|
||||||
enc.GasUsed = hexutil.Uint64(r.GasUsed)
|
enc.GasUsed = hexutil.Uint64(r.GasUsed)
|
||||||
|
enc.BlockHash = r.BlockHash
|
||||||
|
enc.BlockNumber = (*hexutil.Big)(r.BlockNumber)
|
||||||
|
enc.TransactionIndex = hexutil.Uint(r.TransactionIndex)
|
||||||
return json.Marshal(&enc)
|
return json.Marshal(&enc)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -47,6 +54,9 @@ func (r *Receipt) UnmarshalJSON(input []byte) error {
|
|||||||
TxHash *common.Hash `json:"transactionHash" gencodec:"required"`
|
TxHash *common.Hash `json:"transactionHash" gencodec:"required"`
|
||||||
ContractAddress *common.Address `json:"contractAddress"`
|
ContractAddress *common.Address `json:"contractAddress"`
|
||||||
GasUsed *hexutil.Uint64 `json:"gasUsed" gencodec:"required"`
|
GasUsed *hexutil.Uint64 `json:"gasUsed" gencodec:"required"`
|
||||||
|
BlockHash *common.Hash `json:"blockHash,omitempty"`
|
||||||
|
BlockNumber *hexutil.Big `json:"blockNumber,omitempty"`
|
||||||
|
TransactionIndex *hexutil.Uint `json:"transactionIndex"`
|
||||||
}
|
}
|
||||||
var dec Receipt
|
var dec Receipt
|
||||||
if err := json.Unmarshal(input, &dec); err != nil {
|
if err := json.Unmarshal(input, &dec); err != nil {
|
||||||
@ -81,5 +91,14 @@ func (r *Receipt) UnmarshalJSON(input []byte) error {
|
|||||||
return errors.New("missing required field 'gasUsed' for Receipt")
|
return errors.New("missing required field 'gasUsed' for Receipt")
|
||||||
}
|
}
|
||||||
r.GasUsed = uint64(*dec.GasUsed)
|
r.GasUsed = uint64(*dec.GasUsed)
|
||||||
|
if dec.BlockHash != nil {
|
||||||
|
r.BlockHash = *dec.BlockHash
|
||||||
|
}
|
||||||
|
if dec.BlockNumber != nil {
|
||||||
|
r.BlockNumber = (*big.Int)(dec.BlockNumber)
|
||||||
|
}
|
||||||
|
if dec.TransactionIndex != nil {
|
||||||
|
r.TransactionIndex = uint(*dec.TransactionIndex)
|
||||||
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
@ -13,6 +13,7 @@ import (
|
|||||||
|
|
||||||
var _ = (*txdataMarshaling)(nil)
|
var _ = (*txdataMarshaling)(nil)
|
||||||
|
|
||||||
|
// MarshalJSON marshals as JSON.
|
||||||
func (t txdata) MarshalJSON() ([]byte, error) {
|
func (t txdata) MarshalJSON() ([]byte, error) {
|
||||||
type txdata struct {
|
type txdata struct {
|
||||||
AccountNonce hexutil.Uint64 `json:"nonce" gencodec:"required"`
|
AccountNonce hexutil.Uint64 `json:"nonce" gencodec:"required"`
|
||||||
@ -40,6 +41,7 @@ func (t txdata) MarshalJSON() ([]byte, error) {
|
|||||||
return json.Marshal(&enc)
|
return json.Marshal(&enc)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// UnmarshalJSON unmarshals from JSON.
|
||||||
func (t *txdata) UnmarshalJSON(input []byte) error {
|
func (t *txdata) UnmarshalJSON(input []byte) error {
|
||||||
type txdata struct {
|
type txdata struct {
|
||||||
AccountNonce *hexutil.Uint64 `json:"nonce" gencodec:"required"`
|
AccountNonce *hexutil.Uint64 `json:"nonce" gencodec:"required"`
|
||||||
|
@ -20,6 +20,7 @@ import (
|
|||||||
"bytes"
|
"bytes"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
|
"math/big"
|
||||||
"unsafe"
|
"unsafe"
|
||||||
|
|
||||||
"github.com/ethereum/go-ethereum/common"
|
"github.com/ethereum/go-ethereum/common"
|
||||||
@ -44,17 +45,24 @@ const (
|
|||||||
|
|
||||||
// Receipt represents the results of a transaction.
|
// Receipt represents the results of a transaction.
|
||||||
type Receipt struct {
|
type Receipt struct {
|
||||||
// Consensus fields
|
// Consensus fields: These fields are defined by the Yellow Paper
|
||||||
PostState []byte `json:"root"`
|
PostState []byte `json:"root"`
|
||||||
Status uint64 `json:"status"`
|
Status uint64 `json:"status"`
|
||||||
CumulativeGasUsed uint64 `json:"cumulativeGasUsed" gencodec:"required"`
|
CumulativeGasUsed uint64 `json:"cumulativeGasUsed" gencodec:"required"`
|
||||||
Bloom Bloom `json:"logsBloom" gencodec:"required"`
|
Bloom Bloom `json:"logsBloom" gencodec:"required"`
|
||||||
Logs []*Log `json:"logs" gencodec:"required"`
|
Logs []*Log `json:"logs" gencodec:"required"`
|
||||||
|
|
||||||
// Implementation fields (don't reorder!)
|
// Implementation fields: These fields are added by geth when processing a transaction.
|
||||||
|
// They are stored in the chain database.
|
||||||
TxHash common.Hash `json:"transactionHash" gencodec:"required"`
|
TxHash common.Hash `json:"transactionHash" gencodec:"required"`
|
||||||
ContractAddress common.Address `json:"contractAddress"`
|
ContractAddress common.Address `json:"contractAddress"`
|
||||||
GasUsed uint64 `json:"gasUsed" gencodec:"required"`
|
GasUsed uint64 `json:"gasUsed" gencodec:"required"`
|
||||||
|
|
||||||
|
// Inclusion information: These fields provide information about the inclusion of the
|
||||||
|
// transaction corresponding to this receipt.
|
||||||
|
BlockHash common.Hash `json:"blockHash,omitempty"`
|
||||||
|
BlockNumber *big.Int `json:"blockNumber,omitempty"`
|
||||||
|
TransactionIndex uint `json:"transactionIndex"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type receiptMarshaling struct {
|
type receiptMarshaling struct {
|
||||||
@ -62,6 +70,8 @@ type receiptMarshaling struct {
|
|||||||
Status hexutil.Uint64
|
Status hexutil.Uint64
|
||||||
CumulativeGasUsed hexutil.Uint64
|
CumulativeGasUsed hexutil.Uint64
|
||||||
GasUsed hexutil.Uint64
|
GasUsed hexutil.Uint64
|
||||||
|
BlockNumber *hexutil.Big
|
||||||
|
TransactionIndex hexutil.Uint
|
||||||
}
|
}
|
||||||
|
|
||||||
// receiptRLP is the consensus encoding of a receipt.
|
// receiptRLP is the consensus encoding of a receipt.
|
||||||
|
@ -566,6 +566,11 @@ func (w *worker) resultLoop() {
|
|||||||
logs []*types.Log
|
logs []*types.Log
|
||||||
)
|
)
|
||||||
for i, receipt := range task.receipts {
|
for i, receipt := range task.receipts {
|
||||||
|
// add block location fields
|
||||||
|
receipt.BlockHash = hash
|
||||||
|
receipt.BlockNumber = block.Number()
|
||||||
|
receipt.TransactionIndex = uint(i)
|
||||||
|
|
||||||
receipts[i] = new(types.Receipt)
|
receipts[i] = new(types.Receipt)
|
||||||
*receipts[i] = *receipt
|
*receipts[i] = *receipt
|
||||||
// Update the block hash in all logs since it is now available and not when the
|
// Update the block hash in all logs since it is now available and not when the
|
||||||
|
Loading…
Reference in New Issue
Block a user