mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-08 03:51:20 +00:00
08590211e6
* add defer collector.Close * add defer collector.Close
259 lines
7.1 KiB
Go
259 lines
7.1 KiB
Go
package snapshotsync
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"encoding/binary"
|
|
"errors"
|
|
"fmt"
|
|
"os"
|
|
|
|
"github.com/ledgerwatch/erigon/common"
|
|
"github.com/ledgerwatch/erigon/common/dbutils"
|
|
"github.com/ledgerwatch/erigon/common/etl"
|
|
"github.com/ledgerwatch/erigon/core/types"
|
|
"github.com/ledgerwatch/erigon/ethdb"
|
|
"github.com/ledgerwatch/erigon/ethdb/kv"
|
|
"github.com/ledgerwatch/erigon/log"
|
|
"github.com/ledgerwatch/erigon/rlp"
|
|
)
|
|
|
|
func GenerateBodiesSnapshot(ctx context.Context, readTX ethdb.Tx, writeTX ethdb.RwTx, toBlock uint64) error {
|
|
readBodyCursor, err := readTX.Cursor(dbutils.BlockBodyPrefix)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
writeBodyCursor, err := writeTX.RwCursor(dbutils.BlockBodyPrefix)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
writeEthTXCursor, err := writeTX.RwCursor(dbutils.EthTx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
readEthTXCursor, err := readTX.Cursor(dbutils.EthTx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
var expectedBaseTxId uint64
|
|
err = ethdb.Walk(readBodyCursor, []byte{}, 0, func(k, v []byte) (bool, error) {
|
|
if binary.BigEndian.Uint64(k) > toBlock {
|
|
return false, nil
|
|
}
|
|
|
|
canonocalHash, err := readTX.GetOne(dbutils.HeaderCanonicalBucket, dbutils.EncodeBlockNumber(binary.BigEndian.Uint64(k)))
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
if !bytes.Equal(canonocalHash, k[8:]) {
|
|
return true, nil
|
|
}
|
|
bd := &types.BodyForStorage{}
|
|
err = rlp.DecodeBytes(v, bd)
|
|
if err != nil {
|
|
return false, fmt.Errorf("block %s decode err %w", common.Bytes2Hex(k), err)
|
|
}
|
|
baseTxId := bd.BaseTxId
|
|
amount := bd.TxAmount
|
|
|
|
bd.BaseTxId = expectedBaseTxId
|
|
newV, err := rlp.EncodeToBytes(bd)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
err = writeBodyCursor.Append(common.CopyBytes(k), newV)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
|
|
newExpectedTx := expectedBaseTxId
|
|
err = ethdb.Walk(readEthTXCursor, dbutils.EncodeBlockNumber(baseTxId), 0, func(k, v []byte) (bool, error) {
|
|
if newExpectedTx >= expectedBaseTxId+uint64(amount) {
|
|
return false, nil
|
|
}
|
|
err = writeEthTXCursor.Append(dbutils.EncodeBlockNumber(newExpectedTx), common.CopyBytes(v))
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
newExpectedTx++
|
|
return true, nil
|
|
})
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
if newExpectedTx > expectedBaseTxId+uint64(amount) {
|
|
fmt.Println("newExpectedTx > expectedBaseTxId+amount", newExpectedTx, expectedBaseTxId, amount, "block", common.Bytes2Hex(k))
|
|
return false, errors.New("newExpectedTx > expectedBaseTxId+amount")
|
|
}
|
|
expectedBaseTxId += uint64(amount)
|
|
return true, nil
|
|
})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func CreateBodySnapshot(readTx ethdb.Tx, lastBlock uint64, snapshotPath string) error {
|
|
kv, err := kv.NewMDBX().WithBucketsConfig(func(defaultBuckets dbutils.BucketsCfg) dbutils.BucketsCfg {
|
|
return dbutils.BucketsCfg{
|
|
dbutils.BlockBodyPrefix: dbutils.BucketsConfigs[dbutils.BlockBodyPrefix],
|
|
dbutils.EthTx: dbutils.BucketsConfigs[dbutils.EthTx],
|
|
}
|
|
}).Path(snapshotPath).Open()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
defer kv.Close()
|
|
writeTX, err := kv.BeginRw(context.Background())
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer writeTX.Rollback()
|
|
err = GenerateBodiesSnapshot(context.TODO(), readTx, writeTX, lastBlock)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return writeTX.Commit()
|
|
}
|
|
|
|
func OpenBodiesSnapshot(dbPath string) (ethdb.RoKV, error) {
|
|
return kv.NewMDBX().Path(dbPath).WithBucketsConfig(func(defaultBuckets dbutils.BucketsCfg) dbutils.BucketsCfg {
|
|
return dbutils.BucketsCfg{
|
|
dbutils.BlockBodyPrefix: dbutils.BucketsConfigs[dbutils.BlockBodyPrefix],
|
|
dbutils.EthTx: dbutils.BucketsConfigs[dbutils.EthTx],
|
|
}
|
|
}).Readonly().Open()
|
|
}
|
|
|
|
func RemoveBlocksData(db ethdb.RoKV, tx ethdb.RwTx, newSnapshot uint64) (err error) {
|
|
log.Info("Remove blocks data", "to", newSnapshot)
|
|
if _, ok := db.(kv.SnapshotUpdater); !ok {
|
|
return errors.New("db don't implement snapshotUpdater interface")
|
|
}
|
|
bodiesSnapshot := db.(kv.SnapshotUpdater).BodiesSnapshot()
|
|
if bodiesSnapshot == nil {
|
|
log.Info("bodiesSnapshot is empty")
|
|
return nil
|
|
}
|
|
blockBodySnapshotReadTX, err := bodiesSnapshot.BeginRo(context.Background())
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer blockBodySnapshotReadTX.Rollback()
|
|
ethtxSnapshotReadTX, err := blockBodySnapshotReadTX.Cursor(dbutils.EthTx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
lastEthTXSnapshotKey, _, err := ethtxSnapshotReadTX.Last()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
rewriteId := binary.BigEndian.Uint64(lastEthTXSnapshotKey) + 1
|
|
|
|
writeTX := tx.(kv.DBTX).DBTX()
|
|
blockBodyWriteCursor, err := writeTX.RwCursor(dbutils.BlockBodyPrefix)
|
|
if err != nil {
|
|
return fmt.Errorf("get bodies cursor %w", err)
|
|
}
|
|
ethTXWriteCursor, err := writeTX.RwCursor(dbutils.EthTx)
|
|
if err != nil {
|
|
return fmt.Errorf("get ethtx cursor %w", err)
|
|
}
|
|
|
|
logPrefix := "RemoveBlocksData"
|
|
bodiesCollector := etl.NewCollector(os.TempDir(), etl.NewSortableBuffer(etl.BufferOptimalSize))
|
|
defer bodiesCollector.Close(logPrefix)
|
|
ethTXCollector := etl.NewCollector(os.TempDir(), etl.NewSortableBuffer(etl.BufferOptimalSize))
|
|
defer ethTXCollector.Close(logPrefix)
|
|
err = ethdb.Walk(blockBodyWriteCursor, dbutils.BlockBodyKey(0, common.Hash{}), 0, func(k, v []byte) (bool, error) {
|
|
if binary.BigEndian.Uint64(k) > newSnapshot {
|
|
return false, nil
|
|
}
|
|
has, err := blockBodySnapshotReadTX.Has(dbutils.BlockBodyPrefix, k)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
bd := types.BodyForStorage{}
|
|
err = rlp.DecodeBytes(v, &bd)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
|
|
if has {
|
|
innerErr := blockBodyWriteCursor.Delete(k, nil)
|
|
if innerErr != nil {
|
|
return false, fmt.Errorf("remove %v err:%w", common.Bytes2Hex(k), innerErr)
|
|
}
|
|
for i := bd.BaseTxId; i < bd.BaseTxId+uint64(bd.TxAmount); i++ {
|
|
err = ethTXWriteCursor.Delete(dbutils.EncodeBlockNumber(i), nil)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
}
|
|
} else {
|
|
collectKey := common.CopyBytes(k)
|
|
oldBaseTxId := bd.BaseTxId
|
|
bd.BaseTxId = rewriteId
|
|
bodyBytes, err := rlp.EncodeToBytes(bd)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
|
|
if bd.TxAmount > 0 {
|
|
txIdKey := make([]byte, 8)
|
|
binary.BigEndian.PutUint64(txIdKey, oldBaseTxId)
|
|
i := uint32(0)
|
|
|
|
for k, v, err := ethTXWriteCursor.SeekExact(txIdKey); k != nil; k, v, err = ethTXWriteCursor.Next() {
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
|
|
err = ethTXCollector.Collect(dbutils.EncodeBlockNumber(rewriteId+uint64(i)), common.CopyBytes(v))
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
|
|
i++
|
|
if i >= bd.TxAmount {
|
|
break
|
|
}
|
|
}
|
|
}
|
|
//we need to remove it to use snapshot data instead
|
|
for i := oldBaseTxId; i < oldBaseTxId+uint64(bd.TxAmount); i++ {
|
|
err = ethTXWriteCursor.Delete(dbutils.EncodeBlockNumber(i), nil)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
}
|
|
|
|
rewriteId += uint64(bd.TxAmount)
|
|
|
|
err = bodiesCollector.Collect(collectKey, bodyBytes)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
}
|
|
|
|
return true, nil
|
|
})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
err = bodiesCollector.Load("bodies", writeTX, dbutils.BlockBodyPrefix, etl.IdentityLoadFunc, etl.TransformArgs{})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
err = ethTXCollector.Load("ethtx", writeTX, dbutils.EthTx, etl.IdentityLoadFunc, etl.TransformArgs{})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|