mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-15 14:38:20 +00:00
503 lines
16 KiB
Go
503 lines
16 KiB
Go
package powchain
|
|
|
|
import (
|
|
"context"
|
|
"encoding/binary"
|
|
"fmt"
|
|
"math/big"
|
|
"time"
|
|
|
|
"github.com/ethereum/go-ethereum"
|
|
"github.com/ethereum/go-ethereum/accounts/abi/bind"
|
|
"github.com/ethereum/go-ethereum/common"
|
|
gethTypes "github.com/ethereum/go-ethereum/core/types"
|
|
"github.com/pkg/errors"
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/core/feed"
|
|
statefeed "github.com/prysmaticlabs/prysm/beacon-chain/core/feed/state"
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/core/helpers"
|
|
coreState "github.com/prysmaticlabs/prysm/beacon-chain/core/state"
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/state/v1"
|
|
contracts "github.com/prysmaticlabs/prysm/contracts/deposit-contract"
|
|
protodb "github.com/prysmaticlabs/prysm/proto/beacon/db"
|
|
ethpb "github.com/prysmaticlabs/prysm/proto/prysm/v1alpha1"
|
|
"github.com/prysmaticlabs/prysm/shared/bytesutil"
|
|
"github.com/prysmaticlabs/prysm/shared/hashutil"
|
|
"github.com/prysmaticlabs/prysm/shared/params"
|
|
"github.com/sirupsen/logrus"
|
|
)
|
|
|
|
var (
|
|
depositEventSignature = hashutil.HashKeccak256([]byte("DepositEvent(bytes,bytes,bytes,bytes,bytes)"))
|
|
)
|
|
|
|
const eth1DataSavingInterval = 100
|
|
const maxTolerableDifference = 50
|
|
const defaultEth1HeaderReqLimit = uint64(1000)
|
|
const depositlogRequestLimit = 10000
|
|
const additiveFactorMultiplier = 0.10
|
|
const multiplicativeDecreaseDivisor = 2
|
|
|
|
func tooMuchDataRequestedError(err error) bool {
|
|
// this error is only infura specific (other providers might have different error messages)
|
|
return err.Error() == "query returned more than 10000 results"
|
|
}
|
|
|
|
// Eth2GenesisPowchainInfo retrieves the genesis time and eth1 block number of the beacon chain
|
|
// from the deposit contract.
|
|
func (s *Service) Eth2GenesisPowchainInfo() (uint64, *big.Int) {
|
|
return s.chainStartData.GenesisTime, big.NewInt(int64(s.chainStartData.GenesisBlock))
|
|
}
|
|
|
|
// ProcessETH1Block processes the logs from the provided eth1Block.
|
|
func (s *Service) ProcessETH1Block(ctx context.Context, blkNum *big.Int) error {
|
|
query := ethereum.FilterQuery{
|
|
Addresses: []common.Address{
|
|
s.cfg.DepositContract,
|
|
},
|
|
FromBlock: blkNum,
|
|
ToBlock: blkNum,
|
|
}
|
|
logs, err := s.httpLogger.FilterLogs(ctx, query)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
for _, filterLog := range logs {
|
|
// ignore logs that are not of the required block number
|
|
if filterLog.BlockNumber != blkNum.Uint64() {
|
|
continue
|
|
}
|
|
if err := s.ProcessLog(ctx, filterLog); err != nil {
|
|
return errors.Wrap(err, "could not process log")
|
|
}
|
|
}
|
|
if !s.chainStartData.Chainstarted {
|
|
if err := s.checkBlockNumberForChainStart(ctx, blkNum); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// ProcessLog is the main method which handles the processing of all
|
|
// logs from the deposit contract on the ETH1.0 chain.
|
|
func (s *Service) ProcessLog(ctx context.Context, depositLog gethTypes.Log) error {
|
|
s.processingLock.RLock()
|
|
defer s.processingLock.RUnlock()
|
|
// Process logs according to their event signature.
|
|
if depositLog.Topics[0] == depositEventSignature {
|
|
if err := s.ProcessDepositLog(ctx, depositLog); err != nil {
|
|
return errors.Wrap(err, "Could not process deposit log")
|
|
}
|
|
if s.lastReceivedMerkleIndex%eth1DataSavingInterval == 0 {
|
|
return s.savePowchainData(ctx)
|
|
}
|
|
return nil
|
|
}
|
|
log.WithField("signature", fmt.Sprintf("%#x", depositLog.Topics[0])).Debug("Not a valid event signature")
|
|
return nil
|
|
}
|
|
|
|
// ProcessDepositLog processes the log which had been received from
|
|
// the ETH1.0 chain by trying to ascertain which participant deposited
|
|
// in the contract.
|
|
func (s *Service) ProcessDepositLog(ctx context.Context, depositLog gethTypes.Log) error {
|
|
pubkey, withdrawalCredentials, amount, signature, merkleTreeIndex, err := contracts.UnpackDepositLogData(depositLog.Data)
|
|
if err != nil {
|
|
return errors.Wrap(err, "Could not unpack log")
|
|
}
|
|
// If we have already seen this Merkle index, skip processing the log.
|
|
// This can happen sometimes when we receive the same log twice from the
|
|
// ETH1.0 network, and prevents us from updating our trie
|
|
// with the same log twice, causing an inconsistent state root.
|
|
index := int64(binary.LittleEndian.Uint64(merkleTreeIndex))
|
|
if index <= s.lastReceivedMerkleIndex {
|
|
return nil
|
|
}
|
|
|
|
if index != s.lastReceivedMerkleIndex+1 {
|
|
missedDepositLogsCount.Inc()
|
|
return errors.Errorf("received incorrect merkle index: wanted %d but got %d", s.lastReceivedMerkleIndex+1, index)
|
|
}
|
|
s.lastReceivedMerkleIndex = index
|
|
|
|
// We then decode the deposit input in order to create a deposit object
|
|
// we can store in our persistent DB.
|
|
depositData := ðpb.Deposit_Data{
|
|
Amount: bytesutil.FromBytes8(amount),
|
|
PublicKey: pubkey,
|
|
Signature: signature,
|
|
WithdrawalCredentials: withdrawalCredentials,
|
|
}
|
|
|
|
depositHash, err := depositData.HashTreeRoot()
|
|
if err != nil {
|
|
return errors.Wrap(err, "Unable to determine hashed value of deposit")
|
|
}
|
|
|
|
// Defensive check to validate incoming index.
|
|
if s.depositTrie.NumOfItems() != int(index) {
|
|
return errors.Errorf("invalid deposit index received: wanted %d but got %d", s.depositTrie.NumOfItems(), index)
|
|
}
|
|
s.depositTrie.Insert(depositHash[:], int(index))
|
|
|
|
proof, err := s.depositTrie.MerkleProof(int(index))
|
|
if err != nil {
|
|
return errors.Wrap(err, "Unable to generate merkle proof for deposit")
|
|
}
|
|
|
|
deposit := ðpb.Deposit{
|
|
Data: depositData,
|
|
Proof: proof,
|
|
}
|
|
|
|
// We always store all historical deposits in the DB.
|
|
err = s.cfg.DepositCache.InsertDeposit(ctx, deposit, depositLog.BlockNumber, index, s.depositTrie.Root())
|
|
if err != nil {
|
|
return errors.Wrap(err, "unable to insert deposit into cache")
|
|
}
|
|
validData := true
|
|
if !s.chainStartData.Chainstarted {
|
|
s.chainStartData.ChainstartDeposits = append(s.chainStartData.ChainstartDeposits, deposit)
|
|
root := s.depositTrie.Root()
|
|
eth1Data := ðpb.Eth1Data{
|
|
DepositRoot: root[:],
|
|
DepositCount: uint64(len(s.chainStartData.ChainstartDeposits)),
|
|
}
|
|
if err := s.processDeposit(ctx, eth1Data, deposit); err != nil {
|
|
log.Errorf("Invalid deposit processed: %v", err)
|
|
validData = false
|
|
}
|
|
} else {
|
|
s.cfg.DepositCache.InsertPendingDeposit(ctx, deposit, depositLog.BlockNumber, index, s.depositTrie.Root())
|
|
}
|
|
if validData {
|
|
log.WithFields(logrus.Fields{
|
|
"eth1Block": depositLog.BlockNumber,
|
|
"publicKey": fmt.Sprintf("%#x", depositData.PublicKey),
|
|
"merkleTreeIndex": index,
|
|
}).Debug("Deposit registered from deposit contract")
|
|
validDepositsCount.Inc()
|
|
// Notify users what is going on, from time to time.
|
|
if !s.chainStartData.Chainstarted {
|
|
deposits := len(s.chainStartData.ChainstartDeposits)
|
|
if deposits%512 == 0 {
|
|
valCount, err := helpers.ActiveValidatorCount(s.preGenesisState, 0)
|
|
if err != nil {
|
|
log.WithError(err).Error("Could not determine active validator count from pre genesis state")
|
|
}
|
|
log.WithFields(logrus.Fields{
|
|
"deposits": deposits,
|
|
"genesisValidators": valCount,
|
|
}).Info("Processing deposits from Ethereum 1 chain")
|
|
}
|
|
}
|
|
} else {
|
|
log.WithFields(logrus.Fields{
|
|
"eth1Block": depositLog.BlockHash.Hex(),
|
|
"eth1Tx": depositLog.TxHash.Hex(),
|
|
"merkleTreeIndex": index,
|
|
}).Info("Invalid deposit registered in deposit contract")
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// ProcessChainStart processes the log which had been received from
|
|
// the ETH1.0 chain by trying to determine when to start the beacon chain.
|
|
func (s *Service) ProcessChainStart(genesisTime uint64, eth1BlockHash [32]byte, blockNumber *big.Int) {
|
|
s.chainStartData.Chainstarted = true
|
|
s.chainStartData.GenesisBlock = blockNumber.Uint64()
|
|
|
|
chainStartTime := time.Unix(int64(genesisTime), 0)
|
|
|
|
for i := range s.chainStartData.ChainstartDeposits {
|
|
proof, err := s.depositTrie.MerkleProof(i)
|
|
if err != nil {
|
|
log.Errorf("Unable to generate deposit proof %v", err)
|
|
}
|
|
s.chainStartData.ChainstartDeposits[i].Proof = proof
|
|
}
|
|
|
|
root := s.depositTrie.Root()
|
|
s.chainStartData.Eth1Data = ðpb.Eth1Data{
|
|
DepositCount: uint64(len(s.chainStartData.ChainstartDeposits)),
|
|
DepositRoot: root[:],
|
|
BlockHash: eth1BlockHash[:],
|
|
}
|
|
|
|
log.WithFields(logrus.Fields{
|
|
"ChainStartTime": chainStartTime,
|
|
}).Info("Minimum number of validators reached for beacon-chain to start")
|
|
s.cfg.StateNotifier.StateFeed().Send(&feed.Event{
|
|
Type: statefeed.ChainStarted,
|
|
Data: &statefeed.ChainStartedData{
|
|
StartTime: chainStartTime,
|
|
},
|
|
})
|
|
if err := s.savePowchainData(s.ctx); err != nil {
|
|
// continue on, if the save fails as this will get re-saved
|
|
// in the next interval.
|
|
log.Error(err)
|
|
}
|
|
}
|
|
|
|
func (s *Service) createGenesisTime(timeStamp uint64) uint64 {
|
|
// adds in the genesis delay to the eth1 block time
|
|
// on which it was triggered.
|
|
return timeStamp + params.BeaconConfig().GenesisDelay
|
|
}
|
|
|
|
// processPastLogs processes all the past logs from the deposit contract and
|
|
// updates the deposit trie with the data from each individual log.
|
|
func (s *Service) processPastLogs(ctx context.Context) error {
|
|
currentBlockNum := s.latestEth1Data.LastRequestedBlock
|
|
deploymentBlock := params.BeaconNetworkConfig().ContractDeploymentBlock
|
|
// Start from the deployment block if our last requested block
|
|
// is behind it. This is as the deposit logs can only start from the
|
|
// block of the deployment of the deposit contract.
|
|
if deploymentBlock > currentBlockNum {
|
|
currentBlockNum = deploymentBlock
|
|
}
|
|
// To store all blocks.
|
|
headersMap := make(map[uint64]*gethTypes.Header)
|
|
rawLogCount, err := s.depositContractCaller.GetDepositCount(&bind.CallOpts{})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
logCount := binary.LittleEndian.Uint64(rawLogCount)
|
|
|
|
// Batch request the desired headers and store them in a
|
|
// map for quick access.
|
|
requestHeaders := func(startBlk uint64, endBlk uint64) error {
|
|
headers, err := s.batchRequestHeaders(startBlk, endBlk)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
for _, h := range headers {
|
|
if h != nil && h.Number != nil {
|
|
headersMap[h.Number.Uint64()] = h
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
latestFollowHeight, err := s.followBlockHeight(ctx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
batchSize := s.cfg.Eth1HeaderReqLimit
|
|
additiveFactor := uint64(float64(batchSize) * additiveFactorMultiplier)
|
|
|
|
for currentBlockNum < latestFollowHeight {
|
|
start := currentBlockNum
|
|
end := currentBlockNum + batchSize
|
|
// Appropriately bound the request, as we do not
|
|
// want request blocks beyond the current follow distance.
|
|
if end > latestFollowHeight {
|
|
end = latestFollowHeight
|
|
}
|
|
query := ethereum.FilterQuery{
|
|
Addresses: []common.Address{
|
|
s.cfg.DepositContract,
|
|
},
|
|
FromBlock: big.NewInt(int64(start)),
|
|
ToBlock: big.NewInt(int64(end)),
|
|
}
|
|
remainingLogs := logCount - uint64(s.lastReceivedMerkleIndex+1)
|
|
// only change the end block if the remaining logs are below the required log limit.
|
|
// reset our query and end block in this case.
|
|
withinLimit := remainingLogs < depositlogRequestLimit
|
|
aboveFollowHeight := end >= latestFollowHeight
|
|
if withinLimit && aboveFollowHeight {
|
|
query.ToBlock = big.NewInt(int64(latestFollowHeight))
|
|
end = latestFollowHeight
|
|
}
|
|
logs, err := s.httpLogger.FilterLogs(ctx, query)
|
|
if err != nil {
|
|
if tooMuchDataRequestedError(err) {
|
|
if batchSize == 0 {
|
|
return errors.New("batch size is zero")
|
|
}
|
|
|
|
// multiplicative decrease
|
|
batchSize /= multiplicativeDecreaseDivisor
|
|
continue
|
|
}
|
|
return err
|
|
}
|
|
// Only request headers before chainstart to correctly determine
|
|
// genesis.
|
|
if !s.chainStartData.Chainstarted {
|
|
if err := requestHeaders(start, end); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
for _, filterLog := range logs {
|
|
if filterLog.BlockNumber > currentBlockNum {
|
|
if err := s.checkHeaderRange(currentBlockNum, filterLog.BlockNumber-1, headersMap, requestHeaders); err != nil {
|
|
return err
|
|
}
|
|
// set new block number after checking for chainstart for previous block.
|
|
s.latestEth1Data.LastRequestedBlock = currentBlockNum
|
|
currentBlockNum = filterLog.BlockNumber
|
|
}
|
|
if err := s.ProcessLog(ctx, filterLog); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
if err := s.checkHeaderRange(currentBlockNum, end, headersMap, requestHeaders); err != nil {
|
|
return err
|
|
}
|
|
currentBlockNum = end
|
|
|
|
if batchSize < s.cfg.Eth1HeaderReqLimit {
|
|
// update the batchSize with additive increase
|
|
batchSize += additiveFactor
|
|
if batchSize > s.cfg.Eth1HeaderReqLimit {
|
|
batchSize = s.cfg.Eth1HeaderReqLimit
|
|
}
|
|
}
|
|
}
|
|
|
|
s.latestEth1Data.LastRequestedBlock = currentBlockNum
|
|
|
|
c, err := s.cfg.BeaconDB.FinalizedCheckpoint(ctx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
fRoot := bytesutil.ToBytes32(c.Root)
|
|
// Return if no checkpoint exists yet.
|
|
if fRoot == params.BeaconConfig().ZeroHash {
|
|
return nil
|
|
}
|
|
fState, err := s.cfg.StateGen.StateByRoot(ctx, fRoot)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if fState != nil && !fState.IsNil() && fState.Eth1DepositIndex() > 0 {
|
|
s.cfg.DepositCache.PrunePendingDeposits(ctx, int64(fState.Eth1DepositIndex()))
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// requestBatchedHeadersAndLogs requests and processes all the headers and
|
|
// logs from the period last polled to now.
|
|
func (s *Service) requestBatchedHeadersAndLogs(ctx context.Context) error {
|
|
// We request for the nth block behind the current head, in order to have
|
|
// stabilized logs when we retrieve it from the 1.0 chain.
|
|
|
|
requestedBlock, err := s.followBlockHeight(ctx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if requestedBlock > s.latestEth1Data.LastRequestedBlock &&
|
|
requestedBlock-s.latestEth1Data.LastRequestedBlock > maxTolerableDifference {
|
|
log.Infof("Falling back to historical headers and logs sync. Current difference is %d", requestedBlock-s.latestEth1Data.LastRequestedBlock)
|
|
return s.processPastLogs(ctx)
|
|
}
|
|
for i := s.latestEth1Data.LastRequestedBlock + 1; i <= requestedBlock; i++ {
|
|
// Cache eth1 block header here.
|
|
_, err := s.BlockHashByHeight(ctx, big.NewInt(int64(i)))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
err = s.ProcessETH1Block(ctx, big.NewInt(int64(i)))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
s.latestEth1Data.LastRequestedBlock = i
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (s *Service) retrieveBlockHashAndTime(ctx context.Context, blkNum *big.Int) ([32]byte, uint64, error) {
|
|
hash, err := s.BlockHashByHeight(ctx, blkNum)
|
|
if err != nil {
|
|
return [32]byte{}, 0, errors.Wrap(err, "could not get eth1 block hash")
|
|
}
|
|
if hash == [32]byte{} {
|
|
return [32]byte{}, 0, errors.Wrap(err, "got empty block hash")
|
|
}
|
|
timeStamp, err := s.BlockTimeByHeight(ctx, blkNum)
|
|
if err != nil {
|
|
return [32]byte{}, 0, errors.Wrap(err, "could not get block timestamp")
|
|
}
|
|
return hash, timeStamp, nil
|
|
}
|
|
|
|
// checkBlockNumberForChainStart checks the given block number for if chainstart has occurred.
|
|
func (s *Service) checkBlockNumberForChainStart(ctx context.Context, blkNum *big.Int) error {
|
|
hash, timeStamp, err := s.retrieveBlockHashAndTime(ctx, blkNum)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
s.checkForChainstart(hash, blkNum, timeStamp)
|
|
return nil
|
|
}
|
|
|
|
func (s *Service) checkHeaderForChainstart(header *gethTypes.Header) {
|
|
s.checkForChainstart(header.Hash(), header.Number, header.Time)
|
|
}
|
|
|
|
func (s *Service) checkHeaderRange(start, end uint64, headersMap map[uint64]*gethTypes.Header,
|
|
requestHeaders func(uint64, uint64) error) error {
|
|
for i := start; i <= end; i++ {
|
|
if !s.chainStartData.Chainstarted {
|
|
h, ok := headersMap[i]
|
|
if !ok {
|
|
if err := requestHeaders(i, end); err != nil {
|
|
return err
|
|
}
|
|
// Retry this block.
|
|
i--
|
|
continue
|
|
}
|
|
s.checkHeaderForChainstart(h)
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// retrieves the current active validator count and genesis time from
|
|
// the provided block time.
|
|
func (s *Service) currentCountAndTime(blockTime uint64) (uint64, uint64) {
|
|
if s.preGenesisState.NumValidators() == 0 {
|
|
return 0, 0
|
|
}
|
|
valCount, err := helpers.ActiveValidatorCount(s.preGenesisState, 0)
|
|
if err != nil {
|
|
log.WithError(err).Error("Could not determine active validator count from pre genesis state")
|
|
return 0, 0
|
|
}
|
|
return valCount, s.createGenesisTime(blockTime)
|
|
}
|
|
|
|
func (s *Service) checkForChainstart(blockHash [32]byte, blockNumber *big.Int, blockTime uint64) {
|
|
valCount, genesisTime := s.currentCountAndTime(blockTime)
|
|
if valCount == 0 {
|
|
return
|
|
}
|
|
triggered := coreState.IsValidGenesisState(valCount, genesisTime)
|
|
if triggered {
|
|
s.chainStartData.GenesisTime = genesisTime
|
|
s.ProcessChainStart(s.chainStartData.GenesisTime, blockHash, blockNumber)
|
|
}
|
|
}
|
|
|
|
// save all powchain related metadata to disk.
|
|
func (s *Service) savePowchainData(ctx context.Context) error {
|
|
pbState, err := v1.ProtobufBeaconState(s.preGenesisState.InnerStateUnsafe())
|
|
if err != nil {
|
|
return err
|
|
}
|
|
eth1Data := &protodb.ETH1ChainData{
|
|
CurrentEth1Data: s.latestEth1Data,
|
|
ChainstartData: s.chainStartData,
|
|
BeaconState: pbState, // I promise not to mutate it!
|
|
Trie: s.depositTrie.ToProto(),
|
|
DepositContainers: s.cfg.DepositCache.AllDepositContainers(ctx),
|
|
}
|
|
return s.cfg.BeaconDB.SavePowchainData(ctx, eth1Data)
|
|
}
|