mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-18 16:44:12 +00:00
e85d4b76aa
Fixes a nil-pointer dereference in Decode when attempting to call `getTransactionReceipt` / `getBlockReceipts` on blocks containing stateSync events.
240 lines
7.8 KiB
Go
240 lines
7.8 KiB
Go
package rawdb
|
|
|
|
import (
|
|
"bytes"
|
|
"errors"
|
|
"fmt"
|
|
"math/big"
|
|
|
|
"github.com/ledgerwatch/erigon-lib/kv"
|
|
"github.com/ledgerwatch/erigon/common"
|
|
"github.com/ledgerwatch/erigon/common/dbutils"
|
|
"github.com/ledgerwatch/erigon/core/types"
|
|
"github.com/ledgerwatch/erigon/ethdb/cbor"
|
|
"github.com/ledgerwatch/erigon/rlp"
|
|
"github.com/ledgerwatch/log/v3"
|
|
)
|
|
|
|
var (
|
|
// bor receipt key
|
|
borReceiptKey = types.BorReceiptKey
|
|
)
|
|
|
|
// HasBorReceipt verifies the existence of all block receipt belonging
|
|
// to a block.
|
|
func HasBorReceipts(db kv.Has, number uint64) bool {
|
|
if has, err := db.Has(kv.BorReceipts, borReceiptKey(number)); !has || err != nil {
|
|
return false
|
|
}
|
|
return true
|
|
}
|
|
|
|
func ReadRawBorReceipt(db kv.Tx, number uint64) (*types.Receipt, bool, error) {
|
|
data, err := db.GetOne(kv.BorReceipts, borReceiptKey(number))
|
|
if err != nil {
|
|
return nil, false, fmt.Errorf("ReadBorReceipt failed getting bor receipt with blockNumber=%d, err=%s", number, err)
|
|
}
|
|
if len(data) == 0 {
|
|
return nil, false, nil
|
|
}
|
|
|
|
var borReceipt *types.Receipt
|
|
err = cbor.Unmarshal(&borReceipt, bytes.NewReader(data))
|
|
if err == nil {
|
|
return borReceipt, false, nil
|
|
}
|
|
|
|
// Convert the receipts from their storage form to their internal representation
|
|
var borStorageReceipt types.ReceiptForStorage
|
|
if err := rlp.DecodeBytes(data, &borStorageReceipt); err != nil {
|
|
log.Error("Invalid receipt array RLP", "err", err)
|
|
return nil, true, err
|
|
}
|
|
|
|
return (*types.Receipt)(&borStorageReceipt), true, nil
|
|
}
|
|
|
|
func ReadBorReceipt(db kv.Tx, blockHash common.Hash, blockNumber uint64, receipts types.Receipts) (*types.Receipt, error) {
|
|
borReceipt, hasEmbeddedLogs, err := ReadRawBorReceipt(db, blockNumber)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if borReceipt == nil {
|
|
return nil, nil
|
|
}
|
|
|
|
if !hasEmbeddedLogs {
|
|
logsData, err := db.GetOne(kv.Log, dbutils.LogKey(blockNumber, uint32(len(receipts))))
|
|
if err != nil {
|
|
return nil, fmt.Errorf("ReadBorReceipt failed getting bor logs with blockNumber=%d, err=%s", blockNumber, err)
|
|
}
|
|
if logsData != nil {
|
|
var logs types.Logs
|
|
if err = cbor.Unmarshal(&logs, bytes.NewReader(logsData)); err != nil {
|
|
return nil, fmt.Errorf("logs unmarshal failed: %w", err)
|
|
}
|
|
borReceipt.Logs = logs
|
|
}
|
|
}
|
|
|
|
types.DeriveFieldsForBorReceipt(borReceipt, blockHash, blockNumber, receipts)
|
|
|
|
return borReceipt, nil
|
|
}
|
|
|
|
// WriteBorReceipt stores all the bor receipt belonging to a block (storing the state sync recipt and log).
|
|
func WriteBorReceipt(tx kv.RwTx, hash common.Hash, number uint64, borReceipt *types.Receipt) error {
|
|
// Convert the bor receipt into their storage form and serialize them
|
|
buf := bytes.NewBuffer(make([]byte, 0, 1024))
|
|
cbor.Marshal(buf, borReceipt.Logs)
|
|
if err := tx.Append(kv.Log, dbutils.LogKey(number, uint32(borReceipt.TransactionIndex)), buf.Bytes()); err != nil {
|
|
return err
|
|
}
|
|
|
|
buf.Reset()
|
|
err := cbor.Marshal(buf, borReceipt)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
// Store the flattened receipt slice
|
|
if err := tx.Append(kv.BorReceipts, borReceiptKey(number), buf.Bytes()); err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
/*
|
|
// DeleteBorReceipt removes receipt data associated with a block hash.
|
|
func DeleteBorReceipt(tx kv.RwTx, hash common.Hash, number uint64) {
|
|
key := borReceiptKey(number)
|
|
|
|
// we delete Bor Receipt log too
|
|
borReceipt, err := ReadBorReceipt(tx, number)
|
|
if err != nil {
|
|
log.Error("Failted to read bor receipt", "err", err)
|
|
}
|
|
if borReceipt != nil {
|
|
if err := tx.Delete(kv.Log, dbutils.LogKey(number, uint32(borReceipt.TransactionIndex))); err != nil {
|
|
log.Error("Failed to delete bor log", "err", err)
|
|
}
|
|
}
|
|
|
|
if err := tx.Delete(kv.BorReceipts, key); err != nil {
|
|
log.Error("Failed to delete bor receipt", "err", err)
|
|
}
|
|
}
|
|
|
|
// ReadBorTransactionWithBlockHash retrieves a specific bor (fake) transaction by tx hash and block hash, along with
|
|
// its added positional metadata.
|
|
func ReadBorTransactionWithBlockHash(db kv.Tx, borTxHash common.Hash, blockHash common.Hash) (*types.Transaction, common.Hash, uint64, uint64, error) {
|
|
blockNumber, err := ReadTxLookupEntry(db, borTxHash)
|
|
if err != nil {
|
|
return nil, common.Hash{}, 0, 0, err
|
|
}
|
|
if blockNumber == nil {
|
|
return nil, common.Hash{}, 0, 0, errors.New("missing block number")
|
|
}
|
|
|
|
bodyForStorage, err := ReadStorageBody(db, blockHash, *blockNumber)
|
|
if err != nil {
|
|
return nil, common.Hash{}, 0, 0, nil
|
|
}
|
|
|
|
var tx types.Transaction = types.NewBorTransaction()
|
|
return &tx, blockHash, *blockNumber, uint64(bodyForStorage.TxAmount), nil
|
|
}
|
|
*/
|
|
|
|
// ReadBorTransaction returns a specific bor (fake) transaction by txn hash, along with
|
|
// its added positional metadata.
|
|
func ReadBorTransaction(db kv.Tx, borTxHash common.Hash) (types.Transaction, common.Hash, uint64, uint64, error) {
|
|
blockNumber, err := ReadBorTxLookupEntry(db, borTxHash)
|
|
if err != nil {
|
|
return nil, common.Hash{}, 0, 0, err
|
|
}
|
|
if blockNumber == nil {
|
|
return nil, common.Hash{}, 0, 0, errors.New("missing block number")
|
|
}
|
|
|
|
return computeBorTransactionForBlockNumber(db, *blockNumber)
|
|
}
|
|
|
|
func ReadBorTxLookupEntry(db kv.Tx, borTxHash common.Hash) (*uint64, error) {
|
|
blockNumBytes, err := db.GetOne(kv.BorTxLookup, borTxHash.Bytes())
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if blockNumBytes == nil {
|
|
return nil, nil
|
|
}
|
|
|
|
blockNum := (new(big.Int).SetBytes(blockNumBytes)).Uint64()
|
|
return &blockNum, nil
|
|
}
|
|
|
|
// ReadBorTransactionForBlockNumber returns a bor (fake) transaction by block number, along with
|
|
// its added positional metadata.
|
|
func ReadBorTransactionForBlockNumber(db kv.Tx, blockNumber uint64) (types.Transaction, common.Hash, uint64, uint64, error) {
|
|
if !HasBorReceipts(db, blockNumber) {
|
|
return nil, common.Hash{}, 0, 0, nil
|
|
}
|
|
return computeBorTransactionForBlockNumber(db, blockNumber)
|
|
}
|
|
|
|
func computeBorTransactionForBlockNumber(db kv.Tx, blockNumber uint64) (types.Transaction, common.Hash, uint64, uint64, error) {
|
|
blockHash, err := ReadCanonicalHash(db, blockNumber)
|
|
if err != nil {
|
|
return nil, common.Hash{}, 0, 0, err
|
|
}
|
|
if blockHash == (common.Hash{}) {
|
|
return nil, common.Hash{}, 0, 0, errors.New("missing block hash")
|
|
}
|
|
|
|
return computeBorTransactionForBlockNumberAndHash(db, blockNumber, blockHash)
|
|
}
|
|
|
|
// ReadBorTransactionForBlockNumberAndHash returns a bor (fake) transaction by block number and block hash, along with
|
|
// its added positional metadata.
|
|
func ReadBorTransactionForBlockNumberAndHash(db kv.Tx, blockNumber uint64, blockHash common.Hash) (types.Transaction, common.Hash, uint64, uint64, error) {
|
|
if !HasBorReceipts(db, blockNumber) {
|
|
return nil, common.Hash{}, 0, 0, nil
|
|
}
|
|
return computeBorTransactionForBlockNumberAndHash(db, blockNumber, blockHash)
|
|
}
|
|
|
|
func computeBorTransactionForBlockNumberAndHash(db kv.Tx, blockNumber uint64, blockHash common.Hash) (types.Transaction, common.Hash, uint64, uint64, error) {
|
|
bodyForStorage, err := ReadStorageBody(db, blockHash, blockNumber)
|
|
if err != nil {
|
|
return nil, common.Hash{}, 0, 0, err
|
|
}
|
|
|
|
var tx types.Transaction = types.NewBorTransaction()
|
|
return tx, blockHash, blockNumber, uint64(bodyForStorage.TxAmount), nil
|
|
}
|
|
|
|
// ReadBorTransactionForBlock retrieves a specific bor (fake) transaction associated with a block, along with
|
|
// its added positional metadata.
|
|
func ReadBorTransactionForBlock(db kv.Tx, block *types.Block) (types.Transaction, common.Hash, uint64, uint64) {
|
|
if !HasBorReceipts(db, block.NumberU64()) {
|
|
return nil, common.Hash{}, 0, 0
|
|
}
|
|
return computeBorTransactionForBlock(db, block)
|
|
}
|
|
|
|
func computeBorTransactionForBlock(db kv.Tx, block *types.Block) (types.Transaction, common.Hash, uint64, uint64) {
|
|
var tx types.Transaction = types.NewBorTransaction()
|
|
return tx, block.Hash(), block.NumberU64(), uint64(len(block.Transactions()))
|
|
}
|
|
|
|
// TruncateBorReceipts removes all bor receipt for given block number or newer
|
|
func TruncateBorReceipts(db kv.RwTx, number uint64) error {
|
|
if err := db.ForEach(kv.BorReceipts, dbutils.EncodeBlockNumber(number), func(k, _ []byte) error {
|
|
return db.Delete(kv.BorReceipts, k)
|
|
}); err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|