From d1b48b4dad087641d6f62394835fe6de21a848fe Mon Sep 17 00:00:00 2001 From: Francesco4203 Date: Mon, 8 Jul 2024 17:51:50 +0700 Subject: [PATCH] consortium-v2/snapshot: add pruneSnapshotPeriodically pruneSnapshot: delete the nSnapshotsPrune oldest snapshots, keep the latestSnapshotsKeep snapshots pruneSnapshotPeriodically: prune the snapshots at the start of each pruningPeriod --- consensus/consortium/v2/consortium.go | 9 +++++ consensus/consortium/v2/snapshot.go | 48 +++++++++++++++++++++++++++ 2 files changed, 57 insertions(+) diff --git a/consensus/consortium/v2/consortium.go b/consensus/consortium/v2/consortium.go index 86df9ca31b..bd982fbb2d 100644 --- a/consensus/consortium/v2/consortium.go +++ b/consensus/consortium/v2/consortium.go @@ -465,6 +465,7 @@ func (c *Consortium) verifyCascadingFields(chain consensus.ChainHeaderReader, he err = c.verifyValidatorFieldsInExtraData(chain, extraData, header) if err != nil { return err + } if isShillin && extraData.HasFinalityVote == 1 { @@ -629,6 +630,9 @@ func (c *Consortium) snapshot(chain consensus.ChainHeaderReader, number uint64, if err := snap.store(c.db); err != nil { return nil, err } + if err := snap.pruneSnapshotPeriodically(c.db, chain); err != nil { + return nil, err + } log.Info("Stored checkpoint snapshot to disk", "number", number, "hash", hash) figure.NewColorFigure("Welcome to DPOS", "", "green", true).Print() break @@ -688,6 +692,10 @@ func (c *Consortium) snapshot(chain consensus.ChainHeaderReader, number uint64, if err = snap.store(c.db); err != nil { return nil, err } + // Prune the snapshot periodically + if err := snap.pruneSnapshotPeriodically(c.db, chain); err != nil { + return nil, err + } log.Trace("Stored snapshot to disk", "number", snap.Number, "hash", snap.Hash) } log.Trace("Checking snapshot data", "number", snap.Number, "validators", snap.validators()) @@ -1832,6 +1840,7 @@ func (c *Consortium) IsPeriodBlock(chain consensus.ChainHeaderReader, header *ty if c.isTest { return c.testTrippPeriod } + number := header.Number.Uint64() if number%c.config.EpochV2 != 0 || !chain.Config().IsTripp(header.Number) { return false diff --git a/consensus/consortium/v2/snapshot.go b/consensus/consortium/v2/snapshot.go index 5ac3a0d27b..a3a60b1b0b 100644 --- a/consensus/consortium/v2/snapshot.go +++ b/consensus/consortium/v2/snapshot.go @@ -16,10 +16,22 @@ import ( blsCommon "github.com/ethereum/go-ethereum/crypto/bls/common" "github.com/ethereum/go-ethereum/ethdb" "github.com/ethereum/go-ethereum/internal/ethapi" + "github.com/ethereum/go-ethereum/log" "github.com/ethereum/go-ethereum/params" lru "github.com/hashicorp/golang-lru" ) +const ( + blocksPerEpoch = 200 + epochsPerPeriod = 144 +) + +var ( + latestSnapshotsKeep = blocksPerEpoch * epochsPerPeriod * 5 // 5 days + snapshotsToBePruned = epochsPerPeriod * 2 // 2 days + pruningPeriod = blocksPerEpoch * epochsPerPeriod * 1 // every 1 day +) + // Snapshot is the state of the authorization validators at a given point in time. type Snapshot struct { // private fields are not json.Marshalled @@ -112,6 +124,42 @@ func loadSnapshot( return snap, nil } +// snapshot pruning +// delete the nSnapshotsPrune oldest snapshots, keep the latestSnapshotsKeep snapshots +func (s *Snapshot) pruneSnapshot(db ethdb.Database, nSnapshotPrune int, chain consensus.ChainHeaderReader) error { + log.Info("Pruning snapshots at block", "block", s.Number, "nSnapshotPrune", nSnapshotPrune) + // Get block number to start pruning + curBlockNumber := s.Number + curBlockNumber -= curBlockNumber % uint64(blocksPerEpoch) // start of the current epoch + curBlockNumber -= uint64(latestSnapshotsKeep) // start of the oldest epoch to keep + + // delete nSnapshotPrune snapshots starting from curBlockNumber to the older ones + batch := db.NewBatch() + for nSnapshotPrune > 0 { + nSnapshotPrune-- + header := chain.GetHeaderByNumber(curBlockNumber) + if header == nil { + // no more snapshots to prune + break + } + curHash := header.Hash() + if err := batch.Delete(append(rawdb.ConsortiumSnapshotPrefix, curHash[:]...)); err != nil { + return err + } + curBlockNumber -= uint64(blocksPerEpoch) + } + log.Info("Pruned snapshots done") + return batch.Write() +} + +// periodically prune the snapshots at the start of each pruningPeriod +func (s *Snapshot) pruneSnapshotPeriodically(db ethdb.Database, chain consensus.ChainHeaderReader) error { + if s.Number%uint64(pruningPeriod) == 0 { + return s.pruneSnapshot(db, snapshotsToBePruned, chain) + } + return nil +} + // store inserts the snapshot into the database. func (s *Snapshot) store(db ethdb.Database) error { blob, err := json.Marshal(s)