erigon-pulse/core/rawdb/bor_receipts.go
Enrique Jose Avila Asapche a968c7ed5b
cleaned up storage and retrieval of state sync receipts (#5379)
* cleaned up storage and retrieval of state sync receipts

* added ctx to the error and got rid of unused var

* missing one

* small change

* log error

* ops

* read header number

* checking if status is succesful

* nil ptr err
2022-09-27 11:39:58 +01:00

216 lines
7.2 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
}
// ReadBorReceiptRLP retrieves the block receipt belonging to a block in RLP encoding.
func ReadBorReceiptRLP(db kv.Getter, hash common.Hash, number uint64) rlp.RawValue {
data, err := db.GetOne(kv.BorReceipts, borReceiptKey(number))
if err != nil {
log.Error("ReadBorReceiptRLP failed", "err", err)
}
return data
}
// ReadBorReceipt retrieves all the bor block receipts belonging to a block, including
// its correspoinding metadata fields. If it is unable to populate these metadata
// fields then nil is returned.
func ReadBorReceipt(db kv.Tx, number uint64) (*types.Receipt, error) {
// We're deriving many fields from the block body, retrieve beside the receipt
data, err := db.GetOne(kv.BorReceipts, borReceiptKey(number))
if err != nil {
return nil, fmt.Errorf("ReadBorReceipt failed getting bor receipt with blockNumber=%d, err=%s", number, err)
}
if data == nil {
return nil, nil
}
var borReceipt *types.Receipt
if err := rlp.DecodeBytes(data, borReceipt); err != nil {
return nil, err
}
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
}