mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-05 10:32:19 +00:00
270 lines
8.0 KiB
Go
270 lines
8.0 KiB
Go
package antiquary
|
|
|
|
import (
|
|
"context"
|
|
"sync/atomic"
|
|
"time"
|
|
|
|
"github.com/ledgerwatch/erigon-lib/common/datadir"
|
|
"github.com/ledgerwatch/erigon-lib/downloader/snaptype"
|
|
proto_downloader "github.com/ledgerwatch/erigon-lib/gointerfaces/downloader"
|
|
"github.com/ledgerwatch/erigon-lib/kv"
|
|
"github.com/ledgerwatch/erigon/cl/clparams"
|
|
"github.com/ledgerwatch/erigon/cl/persistence"
|
|
"github.com/ledgerwatch/erigon/cl/persistence/beacon_indicies"
|
|
state_accessors "github.com/ledgerwatch/erigon/cl/persistence/state"
|
|
"github.com/ledgerwatch/erigon/cl/phase1/core/state"
|
|
"github.com/ledgerwatch/erigon/cl/utils"
|
|
"github.com/ledgerwatch/erigon/turbo/snapshotsync/freezeblocks"
|
|
"github.com/ledgerwatch/log/v3"
|
|
"github.com/spf13/afero"
|
|
)
|
|
|
|
const safetyMargin = 10_000 // We retire snapshots 10k blocks after the finalized head
|
|
|
|
// Antiquary is where the snapshots go, aka old history, it is what keep track of the oldest records.
|
|
type Antiquary struct {
|
|
mainDB kv.RwDB // this is the main DB
|
|
dirs datadir.Dirs
|
|
downloader proto_downloader.DownloaderClient
|
|
logger log.Logger
|
|
sn *freezeblocks.CaplinSnapshots
|
|
snReader freezeblocks.BeaconSnapshotReader
|
|
ctx context.Context
|
|
beaconDB persistence.BlockSource
|
|
backfilled *atomic.Bool
|
|
cfg *clparams.BeaconChainConfig
|
|
states bool
|
|
fs afero.Fs
|
|
validatorsTable *state_accessors.StaticValidatorTable
|
|
genesisState *state.CachingBeaconState
|
|
// set to nil
|
|
currentState *state.CachingBeaconState
|
|
balances32 []byte
|
|
}
|
|
|
|
func NewAntiquary(ctx context.Context, genesisState *state.CachingBeaconState, validatorsTable *state_accessors.StaticValidatorTable, cfg *clparams.BeaconChainConfig, dirs datadir.Dirs, downloader proto_downloader.DownloaderClient, mainDB kv.RwDB, sn *freezeblocks.CaplinSnapshots, reader freezeblocks.BeaconSnapshotReader, beaconDB persistence.BlockSource, logger log.Logger, states bool, fs afero.Fs) *Antiquary {
|
|
backfilled := &atomic.Bool{}
|
|
backfilled.Store(false)
|
|
return &Antiquary{
|
|
mainDB: mainDB,
|
|
dirs: dirs,
|
|
downloader: downloader,
|
|
logger: logger,
|
|
sn: sn,
|
|
beaconDB: beaconDB,
|
|
ctx: ctx,
|
|
backfilled: backfilled,
|
|
cfg: cfg,
|
|
states: states,
|
|
snReader: reader,
|
|
fs: fs,
|
|
validatorsTable: validatorsTable,
|
|
genesisState: genesisState,
|
|
}
|
|
}
|
|
|
|
// Antiquate is the function that starts transactions seeding and shit, very cool but very shit too as a name.
|
|
func (a *Antiquary) Loop() error {
|
|
if a.downloader == nil {
|
|
return nil // Just skip if we don't have a downloader
|
|
}
|
|
// Skip if we dont support backfilling for the current network
|
|
if !clparams.SupportBackfilling(a.cfg.DepositNetworkID) {
|
|
return nil
|
|
}
|
|
statsReply, err := a.downloader.Stats(a.ctx, &proto_downloader.StatsRequest{})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
reCheckTicker := time.NewTicker(3 * time.Second)
|
|
defer reCheckTicker.Stop()
|
|
// Fist part of the antiquate is to download caplin snapshots
|
|
for !statsReply.Completed {
|
|
select {
|
|
case <-reCheckTicker.C:
|
|
statsReply, err = a.downloader.Stats(a.ctx, &proto_downloader.StatsRequest{})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
case <-a.ctx.Done():
|
|
}
|
|
}
|
|
if err := a.sn.BuildMissingIndices(a.ctx, a.logger, log.LvlDebug); err != nil {
|
|
return err
|
|
}
|
|
// Here we need to start mdbx transaction and lock the thread
|
|
log.Info("[Antiquary]: Stopping Caplin to process historical indicies")
|
|
tx, err := a.mainDB.BeginRw(a.ctx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer tx.Rollback()
|
|
// read the last beacon snapshots
|
|
from, err := beacon_indicies.ReadLastBeaconSnapshot(tx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
logInterval := time.NewTicker(30 * time.Second)
|
|
if err := a.sn.ReopenFolder(); err != nil {
|
|
return err
|
|
}
|
|
defer logInterval.Stop()
|
|
|
|
// Now write the snapshots as indicies
|
|
for i := from; i < a.sn.BlocksAvailable(); i++ {
|
|
// read the snapshot
|
|
header, elBlockNumber, elBlockHash, err := a.sn.ReadHeader(i)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if header == nil {
|
|
continue
|
|
}
|
|
blockRoot, err := header.Header.HashSSZ()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if err := beacon_indicies.MarkRootCanonical(a.ctx, tx, header.Header.Slot, blockRoot); err != nil {
|
|
return err
|
|
}
|
|
if err := beacon_indicies.WriteHeaderSlot(tx, blockRoot, header.Header.Slot); err != nil {
|
|
return err
|
|
}
|
|
if err := beacon_indicies.WriteStateRoot(tx, blockRoot, header.Header.Root); err != nil {
|
|
return err
|
|
}
|
|
if err := beacon_indicies.WriteParentBlockRoot(a.ctx, tx, blockRoot, header.Header.ParentRoot); err != nil {
|
|
return err
|
|
}
|
|
if err := beacon_indicies.WriteExecutionBlockNumber(tx, blockRoot, elBlockNumber); err != nil {
|
|
return err
|
|
}
|
|
if err := beacon_indicies.WriteExecutionBlockHash(tx, blockRoot, elBlockHash); err != nil {
|
|
return err
|
|
}
|
|
select {
|
|
case <-logInterval.C:
|
|
log.Info("[Antiquary]: Processed snapshots", "progress", i, "target", a.sn.BlocksAvailable())
|
|
case <-a.ctx.Done():
|
|
default:
|
|
}
|
|
}
|
|
|
|
frozenSlots := a.sn.BlocksAvailable()
|
|
if frozenSlots != 0 {
|
|
if err := a.beaconDB.PurgeRange(a.ctx, tx, 0, frozenSlots); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
if a.states {
|
|
go a.loopStates(a.ctx)
|
|
}
|
|
|
|
// write the indicies
|
|
if err := beacon_indicies.WriteLastBeaconSnapshot(tx, frozenSlots); err != nil {
|
|
return err
|
|
}
|
|
log.Info("[Antiquary]: Restarting Caplin")
|
|
if err := tx.Commit(); err != nil {
|
|
return err
|
|
}
|
|
// Check for snapshots retirement every 3 minutes
|
|
retirementTicker := time.NewTicker(3 * time.Minute)
|
|
defer retirementTicker.Stop()
|
|
for {
|
|
select {
|
|
case <-retirementTicker.C:
|
|
if !a.backfilled.Load() {
|
|
continue
|
|
}
|
|
var (
|
|
from uint64
|
|
to uint64
|
|
)
|
|
if err := a.mainDB.View(a.ctx, func(roTx kv.Tx) error {
|
|
// read the last beacon snapshots
|
|
from, err = beacon_indicies.ReadLastBeaconSnapshot(roTx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
from += 1
|
|
// read the finalized head
|
|
to, err = beacon_indicies.ReadHighestFinalized(roTx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}); err != nil {
|
|
return err
|
|
}
|
|
// Sanity checks just to be safe.
|
|
if from >= to {
|
|
continue
|
|
}
|
|
to = utils.Min64(to, to-safetyMargin) // We don't want to retire snapshots that are too close to the finalized head
|
|
to = (to / snaptype.Erigon2MergeLimit) * snaptype.Erigon2MergeLimit
|
|
if to-from < snaptype.Erigon2MergeLimit {
|
|
continue
|
|
}
|
|
if err := a.antiquate(from, to); err != nil {
|
|
return err
|
|
}
|
|
case <-a.ctx.Done():
|
|
}
|
|
}
|
|
}
|
|
|
|
// Antiquate will antiquate a specific block range (aka. retire snapshots), this should be ran in the background.
|
|
func (a *Antiquary) antiquate(from, to uint64) error {
|
|
if a.downloader == nil {
|
|
return nil // Just skip if we don't have a downloader
|
|
}
|
|
log.Info("[Antiquary]: Antiquating", "from", from, "to", to)
|
|
if err := freezeblocks.DumpBeaconBlocks(a.ctx, a.mainDB, a.beaconDB, from, to, snaptype.Erigon2MergeLimit, a.dirs.Tmp, a.dirs.Snap, 1, log.LvlDebug, a.logger); err != nil {
|
|
return err
|
|
}
|
|
|
|
roTx, err := a.mainDB.BeginRo(a.ctx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer roTx.Rollback()
|
|
if err := a.beaconDB.PurgeRange(a.ctx, roTx, from, to-from-1); err != nil {
|
|
return err
|
|
}
|
|
roTx.Rollback()
|
|
if err := a.sn.ReopenFolder(); err != nil {
|
|
return err
|
|
}
|
|
|
|
paths := a.sn.SegFilePaths(from, to)
|
|
downloadItems := make([]*proto_downloader.AddItem, len(paths))
|
|
for i, path := range paths {
|
|
downloadItems[i] = &proto_downloader.AddItem{
|
|
Path: path,
|
|
}
|
|
}
|
|
// Notify bittorent to seed the new snapshots
|
|
if _, err := a.downloader.Add(a.ctx, &proto_downloader.AddRequest{Items: downloadItems}); err != nil {
|
|
return err
|
|
}
|
|
|
|
tx, err := a.mainDB.BeginRw(a.ctx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer tx.Rollback()
|
|
a.validatorsTable.SetSlot(to)
|
|
if err := beacon_indicies.WriteLastBeaconSnapshot(tx, to-1); err != nil {
|
|
return err
|
|
}
|
|
return tx.Commit()
|
|
}
|
|
|
|
func (a *Antiquary) NotifyBackfilled() {
|
|
// we set up the range for [lowestRawSlot, finalized]
|
|
a.backfilled.Store(true) // this is the lowest slot not in snapshots
|
|
}
|