mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-20 09:21:11 +00:00
869 lines
31 KiB
Go
869 lines
31 KiB
Go
package antiquary
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"crypto/sha256"
|
|
"encoding/binary"
|
|
"fmt"
|
|
"io"
|
|
"os"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/klauspost/compress/zstd"
|
|
"github.com/ledgerwatch/erigon-lib/common"
|
|
libcommon "github.com/ledgerwatch/erigon-lib/common"
|
|
"github.com/ledgerwatch/erigon-lib/etl"
|
|
"github.com/ledgerwatch/erigon-lib/kv"
|
|
"github.com/ledgerwatch/erigon/cl/clparams"
|
|
"github.com/ledgerwatch/erigon/cl/clparams/initial_state"
|
|
"github.com/ledgerwatch/erigon/cl/cltypes"
|
|
"github.com/ledgerwatch/erigon/cl/cltypes/solid"
|
|
"github.com/ledgerwatch/erigon/cl/persistence/base_encoding"
|
|
"github.com/ledgerwatch/erigon/cl/persistence/beacon_indicies"
|
|
state_accessors "github.com/ledgerwatch/erigon/cl/persistence/state"
|
|
"github.com/ledgerwatch/erigon/cl/persistence/state/historical_states_reader"
|
|
"github.com/ledgerwatch/erigon/cl/phase1/core/state"
|
|
"github.com/ledgerwatch/erigon/cl/phase1/core/state/raw"
|
|
"github.com/ledgerwatch/erigon/cl/phase1/core/state/shuffling"
|
|
"github.com/ledgerwatch/erigon/cl/transition"
|
|
"github.com/ledgerwatch/erigon/cl/transition/impl/eth2"
|
|
"github.com/ledgerwatch/log/v3"
|
|
)
|
|
|
|
// pool for buffers
|
|
var bufferPool = sync.Pool{
|
|
New: func() interface{} {
|
|
return &bytes.Buffer{}
|
|
},
|
|
}
|
|
|
|
func excludeDuplicatesIdentity() etl.LoadFunc {
|
|
var prevKey, prevValue []byte
|
|
prevValue = []byte{}
|
|
return func(k, v []byte, table etl.CurrentTableReader, next etl.LoadNextFunc) error {
|
|
if len(prevKey) == 0 {
|
|
prevKey = common.Copy(k)
|
|
prevValue = common.Copy(v)
|
|
return nil
|
|
}
|
|
if bytes.Equal(k, prevKey) {
|
|
prevValue = common.Copy(v)
|
|
return nil
|
|
}
|
|
if err := next(prevKey, prevKey, prevValue); err != nil {
|
|
return err
|
|
}
|
|
prevKey = common.Copy(k)
|
|
prevValue = common.Copy(v)
|
|
return nil
|
|
}
|
|
}
|
|
|
|
func (s *Antiquary) loopStates(ctx context.Context) {
|
|
// Execute this each second
|
|
reqRetryTimer := time.NewTicker(100 * time.Millisecond)
|
|
defer reqRetryTimer.Stop()
|
|
if !initial_state.IsGenesisStateSupported(clparams.NetworkType(s.cfg.DepositNetworkID)) {
|
|
s.logger.Warn("Genesis state is not supported for this network, no historical states data will be available")
|
|
return
|
|
}
|
|
|
|
for {
|
|
select {
|
|
// Check if we are behind finalized
|
|
case <-reqRetryTimer.C:
|
|
if !s.backfilled.Load() {
|
|
continue
|
|
}
|
|
// Check if we are behind finalized
|
|
progress, finalized, err := s.readHistoricalProcessingProgress(ctx)
|
|
if err != nil {
|
|
s.logger.Error("Failed to read historical processing progress", "err", err)
|
|
continue
|
|
}
|
|
// Stay behind a little bit we rely on forkchoice up until (finalized - 2*slotsPerEpoch)
|
|
if progress+s.cfg.SlotsPerEpoch/2 >= finalized {
|
|
continue
|
|
}
|
|
if err := s.IncrementBeaconState(ctx, finalized); err != nil {
|
|
slot := uint64(0)
|
|
if s.currentState != nil {
|
|
slot = s.currentState.Slot()
|
|
}
|
|
s.logger.Error("Failed to increment beacon state", "err", err, "slot", slot)
|
|
return
|
|
}
|
|
|
|
case <-ctx.Done():
|
|
return
|
|
}
|
|
}
|
|
}
|
|
|
|
func (s *Antiquary) readHistoricalProcessingProgress(ctx context.Context) (progress, finalized uint64, err error) {
|
|
var tx kv.Tx
|
|
tx, err = s.mainDB.BeginRo(ctx)
|
|
if err != nil {
|
|
return
|
|
}
|
|
defer tx.Rollback()
|
|
progress, err = state_accessors.GetStateProcessingProgress(tx)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
finalized, err = beacon_indicies.ReadHighestFinalized(tx)
|
|
if err != nil {
|
|
return
|
|
}
|
|
return
|
|
}
|
|
|
|
func uint64BalancesList(s *state.CachingBeaconState, out []uint64) []uint64 {
|
|
if len(out) < s.ValidatorLength() {
|
|
out = make([]uint64, s.ValidatorLength())
|
|
}
|
|
out = out[:s.ValidatorLength()]
|
|
|
|
s.ForEachBalance(func(v uint64, index int, total int) bool {
|
|
out[index] = v
|
|
return true
|
|
})
|
|
return out
|
|
}
|
|
|
|
func (s *Antiquary) IncrementBeaconState(ctx context.Context, to uint64) error {
|
|
var tx kv.Tx
|
|
|
|
tx, err := s.mainDB.BeginRo(ctx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer tx.Rollback()
|
|
|
|
// maps which validators changes
|
|
changedValidators := make(map[uint64]struct{})
|
|
|
|
loadfunc := func(k, v []byte, table etl.CurrentTableReader, next etl.LoadNextFunc) error {
|
|
return next(k, k, v)
|
|
}
|
|
|
|
etlBufSz := etl.BufferOptimalSize / 8 // 18 collectors * 256mb / 8 = 512mb in worst case
|
|
effectiveBalance := etl.NewCollector(kv.ValidatorEffectiveBalance, s.dirs.Tmp, etl.NewSortableBuffer(etlBufSz), s.logger)
|
|
defer effectiveBalance.Close()
|
|
balances := etl.NewCollector(kv.ValidatorBalance, s.dirs.Tmp, etl.NewSortableBuffer(etlBufSz), s.logger)
|
|
defer balances.Close()
|
|
randaoMixes := etl.NewCollector(kv.RandaoMixes, s.dirs.Tmp, etl.NewSortableBuffer(etlBufSz), s.logger)
|
|
defer randaoMixes.Close()
|
|
intraRandaoMixes := etl.NewCollector(kv.IntraRandaoMixes, s.dirs.Tmp, etl.NewSortableBuffer(etlBufSz), s.logger)
|
|
defer intraRandaoMixes.Close()
|
|
proposers := etl.NewCollector(kv.Proposers, s.dirs.Tmp, etl.NewSortableBuffer(etlBufSz), s.logger)
|
|
defer proposers.Close()
|
|
slashings := etl.NewCollector(kv.ValidatorSlashings, s.dirs.Tmp, etl.NewSortableBuffer(etlBufSz), s.logger)
|
|
defer slashings.Close()
|
|
blockRoots := etl.NewCollector(kv.BlockRoot, s.dirs.Tmp, etl.NewSortableBuffer(etlBufSz), s.logger)
|
|
defer blockRoots.Close()
|
|
stateRoots := etl.NewCollector(kv.StateRoot, s.dirs.Tmp, etl.NewSortableBuffer(etlBufSz), s.logger)
|
|
defer stateRoots.Close()
|
|
slotData := etl.NewCollector(kv.SlotData, s.dirs.Tmp, etl.NewSortableBuffer(etlBufSz), s.logger)
|
|
defer slotData.Close()
|
|
epochData := etl.NewCollector(kv.EpochData, s.dirs.Tmp, etl.NewSortableBuffer(etlBufSz), s.logger)
|
|
defer epochData.Close()
|
|
inactivityScoresC := etl.NewCollector(kv.InactivityScores, s.dirs.Tmp, etl.NewSortableBuffer(etlBufSz), s.logger)
|
|
defer inactivityScoresC.Close()
|
|
nextSyncCommittee := etl.NewCollector(kv.NextSyncCommittee, s.dirs.Tmp, etl.NewSortableBuffer(etlBufSz), s.logger)
|
|
defer nextSyncCommittee.Close()
|
|
currentSyncCommittee := etl.NewCollector(kv.CurrentSyncCommittee, s.dirs.Tmp, etl.NewSortableBuffer(etlBufSz), s.logger)
|
|
defer currentSyncCommittee.Close()
|
|
currentEpochAttestations := etl.NewCollector(kv.CurrentEpochAttestations, s.dirs.Tmp, etl.NewSortableBuffer(etlBufSz), s.logger)
|
|
defer currentEpochAttestations.Close()
|
|
previousEpochAttestations := etl.NewCollector(kv.PreviousEpochAttestations, s.dirs.Tmp, etl.NewSortableBuffer(etlBufSz), s.logger)
|
|
defer previousEpochAttestations.Close()
|
|
eth1DataVotes := etl.NewCollector(kv.Eth1DataVotes, s.dirs.Tmp, etl.NewSortableBuffer(etlBufSz), s.logger)
|
|
defer eth1DataVotes.Close()
|
|
stateEvents := etl.NewCollector(kv.StateEvents, s.dirs.Tmp, etl.NewSortableBuffer(etlBufSz), s.logger)
|
|
defer stateEvents.Close()
|
|
activeValidatorIndicies := etl.NewCollector(kv.ActiveValidatorIndicies, s.dirs.Tmp, etl.NewSortableBuffer(etlBufSz), s.logger)
|
|
defer activeValidatorIndicies.Close()
|
|
|
|
progress, err := state_accessors.GetStateProcessingProgress(tx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
// Go back a little bit
|
|
if progress > s.cfg.SlotsPerEpoch*2 {
|
|
progress -= s.cfg.SlotsPerEpoch * 2
|
|
} else {
|
|
progress = 0
|
|
}
|
|
progress, err = findNearestSlotBackwards(tx, s.cfg, progress) // Maybe the guess was a missed slot.
|
|
if err != nil {
|
|
return err
|
|
}
|
|
// buffers
|
|
commonBuffer := &bytes.Buffer{}
|
|
compressedWriter, err := zstd.NewWriter(commonBuffer, zstd.WithEncoderLevel(zstd.SpeedBetterCompression))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer compressedWriter.Close()
|
|
// TODO(Giulio2002): also store genesis information and resume from state.
|
|
if s.currentState == nil {
|
|
// progress is 0 when we are at genesis
|
|
if progress == 0 {
|
|
s.currentState, err = s.genesisState.Copy()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
// Collect genesis state if we are at genesis
|
|
if err := s.collectGenesisState(ctx, compressedWriter, s.currentState, currentSyncCommittee, nextSyncCommittee, slashings, epochData, inactivityScoresC, proposers, slotData, stateEvents, changedValidators); err != nil {
|
|
return err
|
|
}
|
|
} else {
|
|
start := time.Now()
|
|
// progress not 0? we need to load the state from the DB
|
|
historicalReader := historical_states_reader.NewHistoricalStatesReader(s.cfg, s.snReader, s.validatorsTable, s.fs, s.genesisState)
|
|
s.currentState, err = historicalReader.ReadHistoricalState(ctx, tx, progress)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to read historical state at slot %d: %w", progress, err)
|
|
}
|
|
end := time.Since(start)
|
|
hashRoot, err := s.currentState.HashSSZ()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
log.Info("Recovered Beacon State", "slot", s.currentState.Slot(), "elapsed", end, "root", libcommon.Hash(hashRoot).String())
|
|
if err := s.currentState.InitBeaconState(); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
s.balances32 = s.balances32[:0]
|
|
s.balances32 = append(s.balances32, s.currentState.RawBalances()...)
|
|
}
|
|
|
|
logLvl := log.LvlInfo
|
|
if to-s.currentState.Slot() < 96 {
|
|
logLvl = log.LvlDebug
|
|
}
|
|
start := time.Now()
|
|
|
|
// Use this as the event slot (it will be incremented by 1 each time we process a block)
|
|
slot := s.currentState.Slot() + 1
|
|
|
|
var prevBalances, prevValSet []byte
|
|
events := state_accessors.NewStateEvents()
|
|
slashingOccured := false
|
|
// var validatorStaticState
|
|
// var validatorStaticState map[uint64]*state.ValidatorStatic
|
|
// Setup state events handlers
|
|
s.currentState.SetEvents(raw.Events{
|
|
OnNewSlashingSegment: func(index int, segment uint64) error {
|
|
slashingOccured = true
|
|
return nil
|
|
},
|
|
OnRandaoMixChange: func(index int, mix [32]byte) error {
|
|
return intraRandaoMixes.Collect(base_encoding.Encode64ToBytes4(slot), mix[:])
|
|
},
|
|
OnNewValidator: func(index int, v solid.Validator, balance uint64) error {
|
|
changedValidators[uint64(index)] = struct{}{}
|
|
events.AddValidator(uint64(index), v)
|
|
return s.validatorsTable.AddValidator(v, uint64(index), slot)
|
|
},
|
|
OnNewValidatorActivationEpoch: func(index int, epoch uint64) error {
|
|
changedValidators[uint64(index)] = struct{}{}
|
|
events.ChangeActivationEpoch(uint64(index), epoch)
|
|
return s.validatorsTable.AddActivationEpoch(uint64(index), slot, epoch)
|
|
},
|
|
OnNewValidatorExitEpoch: func(index int, epoch uint64) error {
|
|
changedValidators[uint64(index)] = struct{}{}
|
|
events.ChangeExitEpoch(uint64(index), epoch)
|
|
return s.validatorsTable.AddExitEpoch(uint64(index), slot, epoch)
|
|
},
|
|
OnNewValidatorWithdrawableEpoch: func(index int, epoch uint64) error {
|
|
changedValidators[uint64(index)] = struct{}{}
|
|
events.ChangeWithdrawableEpoch(uint64(index), epoch)
|
|
return s.validatorsTable.AddWithdrawableEpoch(uint64(index), slot, epoch)
|
|
},
|
|
OnNewValidatorSlashed: func(index int, newSlashed bool) error {
|
|
changedValidators[uint64(index)] = struct{}{}
|
|
events.ChangeSlashed(uint64(index), newSlashed)
|
|
return s.validatorsTable.AddSlashed(uint64(index), slot, newSlashed)
|
|
},
|
|
OnNewValidatorActivationEligibilityEpoch: func(index int, epoch uint64) error {
|
|
changedValidators[uint64(index)] = struct{}{}
|
|
events.ChangeActivationEligibilityEpoch(uint64(index), epoch)
|
|
return s.validatorsTable.AddActivationEligibility(uint64(index), slot, epoch)
|
|
},
|
|
OnNewValidatorWithdrawalCredentials: func(index int, wc []byte) error {
|
|
changedValidators[uint64(index)] = struct{}{}
|
|
events.ChangeWithdrawalCredentials(uint64(index), libcommon.BytesToHash(wc))
|
|
return s.validatorsTable.AddWithdrawalCredentials(uint64(index), slot, libcommon.BytesToHash(wc))
|
|
},
|
|
OnEpochBoundary: func(epoch uint64) error {
|
|
if err := s.storeEpochData(commonBuffer, s.currentState, epochData); err != nil {
|
|
return err
|
|
}
|
|
var prevEpoch uint64
|
|
if epoch > 0 {
|
|
prevEpoch = epoch - 1
|
|
}
|
|
mix := s.currentState.GetRandaoMixes(prevEpoch)
|
|
if err := randaoMixes.Collect(base_encoding.Encode64ToBytes4(prevEpoch*s.cfg.SlotsPerEpoch), mix[:]); err != nil {
|
|
return err
|
|
}
|
|
// Write active validator indicies
|
|
actives := s.currentState.GetActiveValidatorsIndices(prevEpoch)
|
|
commonBuffer.Reset()
|
|
if err := base_encoding.WriteRabbits(actives, commonBuffer); err != nil {
|
|
return err
|
|
}
|
|
if err := activeValidatorIndicies.Collect(base_encoding.Encode64ToBytes4(prevEpoch*s.cfg.SlotsPerEpoch), libcommon.Copy(commonBuffer.Bytes())); err != nil {
|
|
return err
|
|
}
|
|
actives = s.currentState.GetActiveValidatorsIndices(epoch)
|
|
commonBuffer.Reset()
|
|
if err := base_encoding.WriteRabbits(actives, commonBuffer); err != nil {
|
|
return err
|
|
}
|
|
if err := activeValidatorIndicies.Collect(base_encoding.Encode64ToBytes4(epoch*s.cfg.SlotsPerEpoch), libcommon.Copy(commonBuffer.Bytes())); err != nil {
|
|
return err
|
|
}
|
|
// truncate the file
|
|
return proposers.Collect(base_encoding.Encode64ToBytes4(epoch), getProposerDutiesValue(s.currentState))
|
|
},
|
|
OnNewBlockRoot: func(index int, root common.Hash) error {
|
|
return blockRoots.Collect(base_encoding.Encode64ToBytes4(s.currentState.Slot()), root[:])
|
|
},
|
|
OnNewStateRoot: func(index int, root common.Hash) error {
|
|
return stateRoots.Collect(base_encoding.Encode64ToBytes4(s.currentState.Slot()), root[:])
|
|
},
|
|
OnNewNextSyncCommittee: func(committee *solid.SyncCommittee) error {
|
|
roundedSlot := s.cfg.RoundSlotToSyncCommitteePeriod(slot)
|
|
return nextSyncCommittee.Collect(base_encoding.Encode64ToBytes4(roundedSlot), committee[:])
|
|
},
|
|
OnNewCurrentSyncCommittee: func(committee *solid.SyncCommittee) error {
|
|
roundedSlot := s.cfg.RoundSlotToSyncCommitteePeriod(slot)
|
|
return currentSyncCommittee.Collect(base_encoding.Encode64ToBytes4(roundedSlot), committee[:])
|
|
},
|
|
OnAppendEth1Data: func(data *cltypes.Eth1Data) error {
|
|
vote, err := data.EncodeSSZ(nil)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return eth1DataVotes.Collect(base_encoding.Encode64ToBytes4(slot), vote)
|
|
},
|
|
})
|
|
log.Log(logLvl, "Starting state processing", "from", slot, "to", to)
|
|
// Set up a timer to log progress
|
|
progressTimer := time.NewTicker(1 * time.Minute)
|
|
defer progressTimer.Stop()
|
|
prevSlot := slot
|
|
first := false
|
|
blocksBeforeCommit := 100_000
|
|
blocksProcessed := 0
|
|
// This tells us that transition and operations do not happen concurrently and access is safe, so we can optimize for GC.
|
|
// there is optimized custom cache to recycle big GC overhead.
|
|
for ; slot < to && blocksProcessed < blocksBeforeCommit; slot++ {
|
|
slashingOccured = false // Set this to false at the beginning of each slot.
|
|
key := base_encoding.Encode64ToBytes4(slot)
|
|
|
|
isDumpSlot := slot%clparams.SlotsPerDump == 0
|
|
block, err := s.snReader.ReadBlockBySlot(ctx, tx, slot)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
prevValidatorSetLength := s.currentState.ValidatorLength()
|
|
prevEpoch := state.Epoch(s.currentState)
|
|
|
|
if slot%s.cfg.SlotsPerEpoch == 0 && s.currentState.Version() == clparams.Phase0Version {
|
|
encoded, err := s.currentState.CurrentEpochAttestations().EncodeSSZ(nil)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if err := s.dumpPayload(base_encoding.Encode64ToBytes4(s.cfg.RoundSlotToEpoch(slot-1)), encoded, currentEpochAttestations, commonBuffer, compressedWriter); err != nil {
|
|
return err
|
|
}
|
|
encoded, err = s.currentState.PreviousEpochAttestations().EncodeSSZ(nil)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if err := s.dumpPayload(base_encoding.Encode64ToBytes4(s.cfg.RoundSlotToEpoch(slot-1)), encoded, previousEpochAttestations, commonBuffer, compressedWriter); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
// If we have a missed block, we just skip it.
|
|
if block == nil {
|
|
if isDumpSlot {
|
|
if err := s.antiquateField(ctx, slot, s.currentState.RawBalances(), compressedWriter, "balances"); err != nil {
|
|
return err
|
|
}
|
|
if err := s.antiquateEffectiveBalances(ctx, slot, s.currentState.RawValidatorSet(), compressedWriter); err != nil {
|
|
return err
|
|
}
|
|
s.balances32 = s.balances32[:0]
|
|
s.balances32 = append(s.balances32, s.currentState.RawBalances()...)
|
|
} else if slot%s.cfg.SlotsPerEpoch == 0 {
|
|
if err := s.antiquateBytesListDiff(ctx, key, s.balances32, s.currentState.RawBalances(), balances, base_encoding.ComputeCompressedSerializedUint64ListDiff); err != nil {
|
|
return err
|
|
}
|
|
s.balances32 = s.balances32[:0]
|
|
s.balances32 = append(s.balances32, s.currentState.RawBalances()...)
|
|
}
|
|
|
|
continue
|
|
}
|
|
// We now compute the difference between the two balances.
|
|
prevBalances = prevBalances[:0]
|
|
prevBalances = append(prevBalances, s.currentState.RawBalances()...)
|
|
prevValSet = prevValSet[:0]
|
|
prevValSet = append(prevValSet, s.currentState.RawValidatorSet()...)
|
|
|
|
fullValidation := slot%1000 == 0 || first
|
|
blockRewardsCollector := ð2.BlockRewardsCollector{}
|
|
// We sanity check the state every 1k slots or when we start.
|
|
if err := transition.TransitionState(s.currentState, block, blockRewardsCollector, fullValidation); err != nil {
|
|
return err
|
|
}
|
|
blocksProcessed++
|
|
|
|
first = false
|
|
|
|
// dump the whole slashings vector.
|
|
if slashingOccured {
|
|
if err := s.antiquateFullUint64List(slashings, slot, s.currentState.RawSlashings(), commonBuffer, compressedWriter); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
if err := s.storeSlotData(commonBuffer, s.currentState, blockRewardsCollector, slotData); err != nil {
|
|
return err
|
|
}
|
|
if err := stateEvents.Collect(base_encoding.Encode64ToBytes4(slot), events.CopyBytes()); err != nil {
|
|
return err
|
|
}
|
|
events.Reset()
|
|
|
|
if isDumpSlot {
|
|
if err := s.antiquateField(ctx, slot, s.currentState.RawBalances(), compressedWriter, "balances"); err != nil {
|
|
return err
|
|
}
|
|
if err := s.antiquateEffectiveBalances(ctx, slot, s.currentState.RawValidatorSet(), compressedWriter); err != nil {
|
|
return err
|
|
}
|
|
// Reset it as we antiquated it.
|
|
s.balances32 = s.balances32[:0]
|
|
s.balances32 = append(s.balances32, s.currentState.RawBalances()...)
|
|
continue
|
|
}
|
|
|
|
// antiquate diffs
|
|
isEpochCrossed := prevEpoch != state.Epoch(s.currentState)
|
|
if slot%s.cfg.SlotsPerEpoch == 0 {
|
|
if err := s.antiquateBytesListDiff(ctx, key, s.balances32, s.currentState.RawBalances(), balances, base_encoding.ComputeCompressedSerializedUint64ListDiff); err != nil {
|
|
return err
|
|
}
|
|
s.balances32 = s.balances32[:0]
|
|
s.balances32 = append(s.balances32, s.currentState.RawBalances()...)
|
|
} else if err := s.antiquateBytesListDiff(ctx, key, prevBalances, s.currentState.RawBalances(), balances, base_encoding.ComputeCompressedSerializedUint64ListDiff); err != nil {
|
|
return err
|
|
}
|
|
if prevValidatorSetLength != s.currentState.ValidatorLength() || isEpochCrossed {
|
|
if err := s.antiquateBytesListDiff(ctx, key, prevValSet, s.currentState.RawValidatorSet(), effectiveBalance, base_encoding.ComputeCompressedSerializedEffectiveBalancesDiff); err != nil {
|
|
return err
|
|
}
|
|
if s.currentState.Version() >= clparams.AltairVersion {
|
|
if err := s.antiquateFullUint64List(inactivityScoresC, slot, s.currentState.RawInactivityScores(), commonBuffer, compressedWriter); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
// We now do some post-processing on the state.
|
|
select {
|
|
case <-progressTimer.C:
|
|
log.Log(logLvl, "State processing progress", "slot", slot, "blk/sec", fmt.Sprintf("%.2f", float64(slot-prevSlot)/60))
|
|
prevSlot = slot
|
|
default:
|
|
}
|
|
}
|
|
tx.Rollback()
|
|
|
|
log.Debug("Finished beacon state iteration", "elapsed", time.Since(start))
|
|
|
|
rwTx, err := s.mainDB.BeginRw(ctx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer rwTx.Rollback()
|
|
start = time.Now()
|
|
log.Log(logLvl, "Stopped Caplin to load states")
|
|
// Now load.
|
|
if err := effectiveBalance.Load(rwTx, kv.ValidatorEffectiveBalance, loadfunc, etl.TransformArgs{Quit: ctx.Done()}); err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := randaoMixes.Load(rwTx, kv.RandaoMixes, loadfunc, etl.TransformArgs{Quit: ctx.Done()}); err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := balances.Load(rwTx, kv.ValidatorBalance, loadfunc, etl.TransformArgs{Quit: ctx.Done()}); err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := proposers.Load(rwTx, kv.Proposers, loadfunc, etl.TransformArgs{Quit: ctx.Done()}); err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := slashings.Load(rwTx, kv.ValidatorSlashings, loadfunc, etl.TransformArgs{Quit: ctx.Done()}); err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := blockRoots.Load(rwTx, kv.BlockRoot, loadfunc, etl.TransformArgs{Quit: ctx.Done()}); err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := stateRoots.Load(rwTx, kv.StateRoot, loadfunc, etl.TransformArgs{Quit: ctx.Done()}); err != nil {
|
|
return err
|
|
}
|
|
if err := activeValidatorIndicies.Load(rwTx, kv.ActiveValidatorIndicies, loadfunc, etl.TransformArgs{Quit: ctx.Done()}); err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := slotData.Load(rwTx, kv.SlotData, loadfunc, etl.TransformArgs{Quit: ctx.Done()}); err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := inactivityScoresC.Load(rwTx, kv.InactivityScores, loadfunc, etl.TransformArgs{Quit: ctx.Done()}); err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := intraRandaoMixes.Load(rwTx, kv.IntraRandaoMixes, loadfunc, etl.TransformArgs{Quit: ctx.Done()}); err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := epochData.Load(rwTx, kv.EpochData, loadfunc, etl.TransformArgs{Quit: ctx.Done()}); err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := nextSyncCommittee.Load(rwTx, kv.NextSyncCommittee, loadfunc, etl.TransformArgs{Quit: ctx.Done()}); err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := currentSyncCommittee.Load(rwTx, kv.CurrentSyncCommittee, loadfunc, etl.TransformArgs{Quit: ctx.Done()}); err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := currentEpochAttestations.Load(rwTx, kv.CurrentEpochAttestations, loadfunc, etl.TransformArgs{Quit: ctx.Done()}); err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := previousEpochAttestations.Load(rwTx, kv.PreviousEpochAttestations, loadfunc, etl.TransformArgs{Quit: ctx.Done()}); err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := eth1DataVotes.Load(rwTx, kv.Eth1DataVotes, loadfunc, etl.TransformArgs{Quit: ctx.Done()}); err != nil {
|
|
return err
|
|
}
|
|
if err := stateEvents.Load(rwTx, kv.StateEvents, loadfunc, etl.TransformArgs{Quit: ctx.Done()}); err != nil {
|
|
return err
|
|
}
|
|
if err := state_accessors.SetStateProcessingProgress(rwTx, s.currentState.Slot()); err != nil {
|
|
return err
|
|
}
|
|
s.validatorsTable.SetSlot(s.currentState.Slot())
|
|
|
|
s.validatorsTable.ForEach(func(validatorIndex uint64, validator *state_accessors.StaticValidator) bool {
|
|
if _, ok := changedValidators[validatorIndex]; !ok {
|
|
return true
|
|
}
|
|
commonBuffer.Reset()
|
|
if err = validator.WriteTo(commonBuffer); err != nil {
|
|
return false
|
|
}
|
|
if err = rwTx.Put(kv.StaticValidators, base_encoding.Encode64ToBytes4(validatorIndex), common.Copy(commonBuffer.Bytes())); err != nil {
|
|
return false
|
|
}
|
|
return true
|
|
})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if err := rwTx.Commit(); err != nil {
|
|
return err
|
|
}
|
|
endTime := time.Since(start)
|
|
stateRoot, err := s.currentState.HashSSZ()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
log.Info("Historical states antiquated", "slot", s.currentState.Slot(), "root", libcommon.Hash(stateRoot), "latency", endTime)
|
|
return nil
|
|
}
|
|
|
|
func (s *Antiquary) antiquateField(ctx context.Context, slot uint64, uncompressed []byte, compressor *zstd.Encoder, name string) error {
|
|
folderPath, filePath := clparams.EpochToPaths(slot, s.cfg, name)
|
|
_ = s.fs.MkdirAll(folderPath, 0o755)
|
|
|
|
balancesFile, err := s.fs.OpenFile(filePath, os.O_TRUNC|os.O_CREATE|os.O_WRONLY, 0644)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer balancesFile.Close()
|
|
compressor.Reset(balancesFile)
|
|
|
|
if err := binary.Write(balancesFile, binary.LittleEndian, uint64(len(uncompressed))); err != nil {
|
|
return err
|
|
}
|
|
|
|
if _, err := compressor.Write(uncompressed); err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := compressor.Close(); err != nil {
|
|
return err
|
|
}
|
|
return balancesFile.Sync()
|
|
}
|
|
|
|
func (s *Antiquary) antiquateEffectiveBalances(ctx context.Context, slot uint64, uncompressed []byte, compressor *zstd.Encoder) error {
|
|
folderPath, filePath := clparams.EpochToPaths(slot, s.cfg, "effective_balances")
|
|
_ = s.fs.MkdirAll(folderPath, 0o755)
|
|
|
|
balancesFile, err := s.fs.OpenFile(filePath, os.O_TRUNC|os.O_CREATE|os.O_WRONLY, 0644)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer balancesFile.Close()
|
|
compressor.Reset(balancesFile)
|
|
validatorSetSize := 121
|
|
|
|
if err := binary.Write(balancesFile, binary.LittleEndian, uint64((len(uncompressed)/validatorSetSize)*8)); err != nil {
|
|
return err
|
|
}
|
|
|
|
for i := 0; i < len(uncompressed)/validatorSetSize; i++ {
|
|
// 80:88
|
|
if _, err := compressor.Write(uncompressed[i*validatorSetSize+80 : i*validatorSetSize+88]); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
if err := compressor.Close(); err != nil {
|
|
return err
|
|
}
|
|
return balancesFile.Sync()
|
|
}
|
|
|
|
func (s *Antiquary) antiquateBytesListDiff(ctx context.Context, key []byte, old, new []byte, collector *etl.Collector, diffFn func(w io.Writer, old, new []byte) error) error {
|
|
// create a diff
|
|
diffBuffer := bufferPool.Get().(*bytes.Buffer)
|
|
defer bufferPool.Put(diffBuffer)
|
|
diffBuffer.Reset()
|
|
|
|
if err := diffFn(diffBuffer, old, new); err != nil {
|
|
return err
|
|
}
|
|
|
|
return collector.Collect(key, common.Copy(diffBuffer.Bytes()))
|
|
}
|
|
|
|
func getProposerDutiesValue(s *state.CachingBeaconState) []byte {
|
|
epoch := state.Epoch(s)
|
|
var wg sync.WaitGroup
|
|
list := make([]byte, s.BeaconConfig().SlotsPerEpoch*4)
|
|
for slot := s.Slot(); slot < s.Slot()+s.BeaconConfig().SlotsPerEpoch; slot++ {
|
|
var proposerIndex uint64
|
|
// Lets do proposer index computation
|
|
mixPosition := (epoch + s.BeaconConfig().EpochsPerHistoricalVector - s.BeaconConfig().MinSeedLookahead - 1) %
|
|
s.BeaconConfig().EpochsPerHistoricalVector
|
|
// Input for the seed hash.
|
|
mix := s.GetRandaoMix(int(mixPosition))
|
|
input := shuffling.GetSeed(s.BeaconConfig(), mix, epoch, s.BeaconConfig().DomainBeaconProposer)
|
|
slotByteArray := make([]byte, 8)
|
|
binary.LittleEndian.PutUint64(slotByteArray, slot)
|
|
|
|
// Add slot to the end of the input.
|
|
inputWithSlot := append(input[:], slotByteArray...)
|
|
hash := sha256.New()
|
|
|
|
// Calculate the hash.
|
|
hash.Write(inputWithSlot)
|
|
seed := hash.Sum(nil)
|
|
|
|
indices := s.GetActiveValidatorsIndices(epoch)
|
|
|
|
// Write the seed to an array.
|
|
seedArray := [32]byte{}
|
|
copy(seedArray[:], seed)
|
|
wg.Add(1)
|
|
|
|
// Do it in parallel
|
|
go func(i, slot uint64, indicies []uint64, seedArray [32]byte) {
|
|
defer wg.Done()
|
|
var err error
|
|
proposerIndex, err = shuffling.ComputeProposerIndex(s.BeaconState, indices, seedArray)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
binary.BigEndian.PutUint32(list[i*4:(i+1)*4], uint32(proposerIndex))
|
|
}(slot-s.Slot(), slot, indices, seedArray)
|
|
}
|
|
wg.Wait()
|
|
return list
|
|
}
|
|
|
|
func (s *Antiquary) collectGenesisState(ctx context.Context, compressor *zstd.Encoder, state *state.CachingBeaconState, currentSyncCommittee, nextSyncCommittee, slashings, epochData, inactivities, proposersCollector, slotDataCollector, stateEvents *etl.Collector, changedValidators map[uint64]struct{}) error {
|
|
var err error
|
|
slot := state.Slot()
|
|
epoch := slot / s.cfg.SlotsPerEpoch
|
|
// Setup state events handlers
|
|
if err := proposersCollector.Collect(base_encoding.Encode64ToBytes4(epoch), getProposerDutiesValue(s.currentState)); err != nil {
|
|
return err
|
|
}
|
|
|
|
events := state_accessors.NewStateEvents()
|
|
|
|
state.ForEachValidator(func(v solid.Validator, index, total int) bool {
|
|
changedValidators[uint64(index)] = struct{}{}
|
|
if err = s.validatorsTable.AddValidator(v, uint64(index), 0); err != nil {
|
|
return false
|
|
}
|
|
events.AddValidator(uint64(index), v)
|
|
return true
|
|
})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
roundedSlotToDump := slot - (slot % clparams.SlotsPerDump)
|
|
if err := s.antiquateField(ctx, roundedSlotToDump, s.currentState.RawBalances(), compressor, "balances"); err != nil {
|
|
return err
|
|
}
|
|
if err := s.antiquateEffectiveBalances(ctx, roundedSlotToDump, s.currentState.RawValidatorSet(), compressor); err != nil {
|
|
return err
|
|
}
|
|
var commonBuffer bytes.Buffer
|
|
if err := s.antiquateFullUint64List(slashings, roundedSlotToDump, s.currentState.RawSlashings(), &commonBuffer, compressor); err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := s.storeEpochData(&commonBuffer, state, epochData); err != nil {
|
|
return err
|
|
}
|
|
|
|
if state.Version() >= clparams.AltairVersion {
|
|
// dump inactivity scores
|
|
if err := s.antiquateFullUint64List(inactivities, slot, state.RawInactivityScores(), &commonBuffer, compressor); err != nil {
|
|
return err
|
|
}
|
|
committeeSlot := s.cfg.RoundSlotToSyncCommitteePeriod(slot)
|
|
committee := *state.CurrentSyncCommittee()
|
|
if err := currentSyncCommittee.Collect(base_encoding.Encode64ToBytes4(committeeSlot), libcommon.Copy(committee[:])); err != nil {
|
|
return err
|
|
}
|
|
|
|
committee = *state.NextSyncCommittee()
|
|
if err := nextSyncCommittee.Collect(base_encoding.Encode64ToBytes4(committeeSlot), libcommon.Copy(committee[:])); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
var b bytes.Buffer
|
|
if err := s.storeSlotData(&b, state, nil, slotDataCollector); err != nil {
|
|
return err
|
|
}
|
|
|
|
return stateEvents.Collect(base_encoding.Encode64ToBytes4(slot), events.CopyBytes())
|
|
}
|
|
|
|
func (s *Antiquary) storeSlotData(buffer *bytes.Buffer, st *state.CachingBeaconState, rewardsCollector *eth2.BlockRewardsCollector, collector *etl.Collector) error {
|
|
buffer.Reset()
|
|
slotData := state_accessors.SlotDataFromBeaconState(st)
|
|
if rewardsCollector != nil {
|
|
slotData.AttestationsRewards = rewardsCollector.Attestations
|
|
slotData.SyncAggregateRewards = rewardsCollector.SyncAggregate
|
|
slotData.AttesterSlashings = rewardsCollector.AttesterSlashings
|
|
slotData.ProposerSlashings = rewardsCollector.ProposerSlashings
|
|
}
|
|
if err := slotData.WriteTo(buffer); err != nil {
|
|
return err
|
|
}
|
|
return collector.Collect(base_encoding.Encode64ToBytes4(st.Slot()), libcommon.Copy(buffer.Bytes()))
|
|
}
|
|
|
|
func (s *Antiquary) storeEpochData(buffer *bytes.Buffer, st *state.CachingBeaconState, collector *etl.Collector) error {
|
|
buffer.Reset()
|
|
epochData := state_accessors.EpochDataFromBeaconState(st)
|
|
|
|
if err := epochData.WriteTo(buffer); err != nil {
|
|
return err
|
|
}
|
|
roundedSlot := s.cfg.RoundSlotToEpoch(st.Slot())
|
|
return collector.Collect(base_encoding.Encode64ToBytes4(roundedSlot), libcommon.Copy(buffer.Bytes()))
|
|
}
|
|
|
|
func (s *Antiquary) dumpPayload(k []byte, v []byte, c *etl.Collector, b *bytes.Buffer, compressor *zstd.Encoder) error {
|
|
if compressor == nil {
|
|
return c.Collect(k, v)
|
|
}
|
|
b.Reset()
|
|
compressor.Reset(b)
|
|
|
|
if _, err := compressor.Write(v); err != nil {
|
|
return err
|
|
}
|
|
if err := compressor.Close(); err != nil {
|
|
return err
|
|
}
|
|
return c.Collect(k, common.Copy(b.Bytes()))
|
|
}
|
|
|
|
// func (s *Antiquary) dumpFullBeaconState() {
|
|
// b, err := s.currentState.EncodeSSZ(nil)
|
|
// if err != nil {
|
|
// s.logger.Error("Failed to encode full beacon state", "err", err)
|
|
// return
|
|
// }
|
|
// // just dump it in a.txt like an idiot without afero
|
|
// if err := os.WriteFile("b.txt", b, 0644); err != nil {
|
|
// s.logger.Error("Failed to write full beacon state", "err", err)
|
|
// }
|
|
// }
|
|
|
|
func flattenRandaoMixes(hashes []libcommon.Hash) []byte {
|
|
out := make([]byte, len(hashes)*32)
|
|
for i, h := range hashes {
|
|
copy(out[i*32:(i+1)*32], h[:])
|
|
}
|
|
return out
|
|
}
|
|
|
|
// antiquateFullSlashings goes on mdbx as it is full of common repeated patter always and thus fits with 16KB pages.
|
|
func (s *Antiquary) antiquateFullUint64List(collector *etl.Collector, slot uint64, raw []byte, buffer *bytes.Buffer, compressor *zstd.Encoder) error {
|
|
buffer.Reset()
|
|
compressor.Reset(buffer)
|
|
if _, err := compressor.Write(raw); err != nil {
|
|
return err
|
|
}
|
|
if err := compressor.Close(); err != nil {
|
|
return err
|
|
}
|
|
return collector.Collect(base_encoding.Encode64ToBytes4(slot), common.Copy(buffer.Bytes()))
|
|
}
|
|
|
|
func findNearestSlotBackwards(tx kv.Tx, cfg *clparams.BeaconChainConfig, slot uint64) (uint64, error) {
|
|
canonicalRoot, err := beacon_indicies.ReadCanonicalBlockRoot(tx, slot)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
for (canonicalRoot == (common.Hash{}) && slot > 0) || slot%cfg.SlotsPerEpoch != 0 {
|
|
slot--
|
|
canonicalRoot, err = beacon_indicies.ReadCanonicalBlockRoot(tx, slot)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
}
|
|
return slot, nil
|
|
}
|