package execution import ( "context" "encoding/binary" "fmt" "math/big" "strings" "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/v5/beacon-chain/cache/depositsnapshot" "github.com/prysmaticlabs/prysm/v5/beacon-chain/core/feed" statefeed "github.com/prysmaticlabs/prysm/v5/beacon-chain/core/feed/state" "github.com/prysmaticlabs/prysm/v5/beacon-chain/core/helpers" coreState "github.com/prysmaticlabs/prysm/v5/beacon-chain/core/transition" "github.com/prysmaticlabs/prysm/v5/beacon-chain/execution/types" statenative "github.com/prysmaticlabs/prysm/v5/beacon-chain/state/state-native" "github.com/prysmaticlabs/prysm/v5/config/features" "github.com/prysmaticlabs/prysm/v5/config/params" "github.com/prysmaticlabs/prysm/v5/container/trie" contracts "github.com/prysmaticlabs/prysm/v5/contracts/deposit" "github.com/prysmaticlabs/prysm/v5/crypto/hash" "github.com/prysmaticlabs/prysm/v5/encoding/bytesutil" ethpb "github.com/prysmaticlabs/prysm/v5/proto/prysm/v1alpha1" "github.com/prysmaticlabs/prysm/v5/time/slots" "github.com/sirupsen/logrus" ) var ( depositEventSignature = hash.Keccak256([]byte("DepositEvent(bytes,bytes,bytes,bytes,bytes)")) ) const eth1DataSavingInterval = 1000 const maxTolerableDifference = 50 const defaultEth1HeaderReqLimit = uint64(1000) const depositLogRequestLimit = 10000 const additiveFactorMultiplier = 0.10 const multiplicativeDecreaseDivisor = 2 const depositLoggingInterval = 1024 var errTimedOut = errors.New("net/http: request canceled") 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" } func clientTimedOutError(err error) bool { return strings.Contains(err.Error(), errTimedOut.Error()) } // GenesisExecutionChainInfo retrieves the genesis time and execution block number of the beacon chain // from the deposit contract. func (s *Service) GenesisExecutionChainInfo() (uint64, *big.Int) { return s.chainStartData.GenesisTime, big.NewInt(int64(s.chainStartData.GenesisBlock)) } // ProcessETH1Block processes logs from the provided eth1 block. func (s *Service) ProcessETH1Block(ctx context.Context, blkNum *big.Int) error { query := ethereum.FilterQuery{ Addresses: []common.Address{ s.cfg.depositContractAddr, }, FromBlock: blkNum, ToBlock: blkNum, } logs, err := s.httpLogger.FilterLogs(ctx, query) if err != nil { return err } for i, filterLog := range logs { // ignore logs that are not of the required block number if filterLog.BlockNumber != blkNum.Uint64() { continue } if err := s.ProcessLog(ctx, &logs[i]); err != nil { return errors.Wrap(err, "could not process log") } } if !s.chainStartData.Chainstarted { if err := s.processChainStartFromBlockNum(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 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 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)) // lint:ignore uintcast -- MerkleTreeIndex should not exceed int64 in your lifetime. 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) } if err = s.depositTrie.Insert(depositHash[:], int(index)); err != nil { return err } deposit := ðpb.Deposit{ Data: depositData, } // Only generate the proofs during pre-genesis. if !s.chainStartData.Chainstarted { proof, err := s.depositTrie.MerkleProof(int(index)) if err != nil { return errors.Wrap(err, "unable to generate merkle proof for deposit") } deposit.Proof = proof } // We always store all historical deposits in the DB. root, err := s.depositTrie.HashTreeRoot() if err != nil { return errors.Wrap(err, "unable to determine root of deposit trie") } err = s.cfg.depositCache.InsertDeposit(ctx, deposit, depositLog.BlockNumber, index, 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, err := s.depositTrie.HashTreeRoot() if err != nil { return errors.Wrap(err, "unable to determine root of deposit trie") } eth1Data := ðpb.Eth1Data{ DepositRoot: root[:], DepositCount: uint64(len(s.chainStartData.ChainstartDeposits)), } if err := s.processDeposit(ctx, eth1Data, deposit); err != nil { log.WithError(err).Error("Invalid deposit processed") validData = false } } else { root, err := s.depositTrie.HashTreeRoot() if err != nil { return errors.Wrap(err, "unable to determine root of deposit trie") } s.cfg.depositCache.InsertPendingDeposit(ctx, deposit, depositLog.BlockNumber, index, root) } if validData { // Log the deposit received periodically if index%depositLoggingInterval == 0 { 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%depositLoggingInterval == 0 { valCount, err := helpers.ActiveValidatorCount(ctx, 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") } if features.Get().EnableEIP4881 { // We finalize the trie here so that old deposits are not kept around, as they make // deposit tree htr computation expensive. dTrie, ok := s.depositTrie.(*depositsnapshot.DepositTree) if !ok { return errors.Errorf("wrong trie type initialized: %T", dTrie) } if err := dTrie.Finalize(index, depositLog.BlockHash, depositLog.BlockNumber); err != nil { log.WithError(err).Error("Could not finalize trie") } } return nil } // ProcessChainStart processes the log which had been received from // the eth1 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) // lint:ignore uintcast -- Genesis time won't exceed int64 in your lifetime. for i := range s.chainStartData.ChainstartDeposits { proof, err := s.depositTrie.MerkleProof(i) if err != nil { log.WithError(err).Error("unable to generate deposit proof") } s.chainStartData.ChainstartDeposits[i].Proof = proof } root, err := s.depositTrie.HashTreeRoot() if err != nil { // This should never happen. log.WithError(err).Error("unable to determine root of deposit trie, aborting chain start") return } 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) } } // createGenesisTime adds in the genesis delay to the eth1 block time // on which it was triggered. func createGenesisTime(timeStamp uint64) uint64 { 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. currentBlockNum = max(currentBlockNum, deploymentBlock) // To store all blocks. headersMap := make(map[uint64]*types.HeaderInfo) rawLogCount, err := s.depositContractCaller.GetDepositCount(&bind.CallOpts{}) if err != nil { return err } logCount := binary.LittleEndian.Uint64(rawLogCount) latestFollowHeight, err := s.followedBlockHeight(ctx) if err != nil { return err } batchSize := s.cfg.eth1HeaderReqLimit additiveFactor := uint64(float64(batchSize) * additiveFactorMultiplier) log.WithFields(logrus.Fields{ "currentEth1Block": latestFollowHeight, "currentLogCount": logCount, }).Debug("Processing historical deposit logs") for currentBlockNum < latestFollowHeight { currentBlockNum, batchSize, err = s.processBlockInBatch(ctx, currentBlockNum, latestFollowHeight, batchSize, additiveFactor, logCount, headersMap) if err != nil { return err } } s.latestEth1DataLock.Lock() s.latestEth1Data.LastRequestedBlock = currentBlockNum s.latestEth1DataLock.Unlock() 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 := s.cfg.finalizedStateAtStartup isNil := fState == nil || fState.IsNil() // If processing past logs take a long time, we // need to check if this is the correct finalized // state we are referring to and whether our cached // finalized state is referring to our current finalized checkpoint. // The current code does ignore an edge case where the finalized // block is in a different epoch from the checkpoint's epoch. // This only happens in skipped slots, so pruning it is not an issue. if isNil || slots.ToEpoch(fState.Slot()) != c.Epoch { 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())) // lint:ignore uintcast -- deposit index should not exceed int64 in your lifetime. } return nil } func (s *Service) processBlockInBatch(ctx context.Context, currentBlockNum uint64, latestFollowHeight uint64, batchSize uint64, additiveFactor uint64, logCount uint64, headersMap map[uint64]*types.HeaderInfo) (uint64, uint64, error) { // 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 } start := currentBlockNum end := currentBlockNum + batchSize // Appropriately bound the request, as we do not // want request blocks beyond the current follow distance. end = min(end, latestFollowHeight) query := ethereum.FilterQuery{ Addresses: []common.Address{ s.cfg.depositContractAddr, }, FromBlock: new(big.Int).SetUint64(start), ToBlock: new(big.Int).SetUint64(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 = new(big.Int).SetUint64(latestFollowHeight) end = latestFollowHeight } logs, err := s.httpLogger.FilterLogs(ctx, query) if err != nil { if tooMuchDataRequestedError(err) { if batchSize == 0 { return 0, 0, errors.New("batch size is zero") } // multiplicative decrease batchSize /= multiplicativeDecreaseDivisor return currentBlockNum, batchSize, nil } return 0, 0, err } // Only request headers before chainstart to correctly determine // genesis. if !s.chainStartData.Chainstarted { if err := requestHeaders(start, end); err != nil { return 0, 0, err } } s.latestEth1DataLock.RLock() lastReqBlock := s.latestEth1Data.LastRequestedBlock s.latestEth1DataLock.RUnlock() for i, filterLog := range logs { if filterLog.BlockNumber > currentBlockNum { if err := s.checkHeaderRange(ctx, currentBlockNum, filterLog.BlockNumber-1, headersMap, requestHeaders); err != nil { return 0, 0, err } // set new block number after checking for chainstart for previous block. s.latestEth1DataLock.Lock() s.latestEth1Data.LastRequestedBlock = currentBlockNum s.latestEth1DataLock.Unlock() currentBlockNum = filterLog.BlockNumber } if err := s.ProcessLog(ctx, &logs[i]); err != nil { // In the event the execution client gives us a garbled/bad log // we reset the last requested block to the previous valid block range. This // prevents the beacon from advancing processing of logs to another range // in the event of an execution client failure. s.latestEth1DataLock.Lock() s.latestEth1Data.LastRequestedBlock = lastReqBlock s.latestEth1DataLock.Unlock() return 0, 0, err } } if err := s.checkHeaderRange(ctx, currentBlockNum, end, headersMap, requestHeaders); err != nil { return 0, 0, 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 } } return currentBlockNum, batchSize, 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 eth1 chain. requestedBlock, err := s.followedBlockHeight(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, new(big.Int).SetUint64(i)) if err != nil { return err } err = s.ProcessETH1Block(ctx, new(big.Int).SetUint64(i)) if err != nil { return err } s.latestEth1DataLock.Lock() s.latestEth1Data.LastRequestedBlock = i s.latestEth1DataLock.Unlock() } return nil } func (s *Service) retrieveBlockHashAndTime(ctx context.Context, blkNum *big.Int) ([32]byte, uint64, error) { bHash, err := s.BlockHashByHeight(ctx, blkNum) if err != nil { return [32]byte{}, 0, errors.Wrap(err, "could not get eth1 block hash") } if bHash == [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 bHash, timeStamp, nil } func (s *Service) processChainStartFromBlockNum(ctx context.Context, blkNum *big.Int) error { bHash, timeStamp, err := s.retrieveBlockHashAndTime(ctx, blkNum) if err != nil { return err } s.processChainStartIfReady(ctx, bHash, blkNum, timeStamp) return nil } func (s *Service) processChainStartFromHeader(ctx context.Context, header *types.HeaderInfo) { s.processChainStartIfReady(ctx, header.Hash, header.Number, header.Time) } func (s *Service) checkHeaderRange(ctx context.Context, start, end uint64, headersMap map[uint64]*types.HeaderInfo, 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.processChainStartFromHeader(ctx, h) } } return nil } // retrieves the current active validator count and genesis time from // the provided block time. func (s *Service) currentCountAndTime(ctx context.Context, blockTime uint64) (uint64, uint64) { if s.preGenesisState.NumValidators() == 0 { return 0, 0 } valCount, err := helpers.ActiveValidatorCount(ctx, 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, createGenesisTime(blockTime) } func (s *Service) processChainStartIfReady(ctx context.Context, blockHash [32]byte, blockNumber *big.Int, blockTime uint64) { valCount, genesisTime := s.currentCountAndTime(ctx, blockTime) if valCount == 0 { return } triggered := coreState.IsValidGenesisState(valCount, genesisTime) if triggered { s.chainStartData.GenesisTime = genesisTime s.ProcessChainStart(s.chainStartData.GenesisTime, blockHash, blockNumber) } } // savePowchainData saves all powchain related metadata to disk. func (s *Service) savePowchainData(ctx context.Context) error { pbState, err := statenative.ProtobufBeaconStatePhase0(s.preGenesisState.ToProtoUnsafe()) if err != nil { return err } eth1Data := ðpb.ETH1ChainData{ CurrentEth1Data: s.latestEth1Data, ChainstartData: s.chainStartData, BeaconState: pbState, // I promise not to mutate it! DepositContainers: s.cfg.depositCache.AllDepositContainers(ctx), } if features.Get().EnableEIP4881 { fd, err := s.cfg.depositCache.FinalizedDeposits(ctx) if err != nil { return errors.Errorf("could not get finalized deposit tree: %v", err) } tree, ok := fd.Deposits().(*depositsnapshot.DepositTree) if !ok { return errors.New("deposit tree was not EIP4881 DepositTree") } eth1Data.DepositSnapshot, err = tree.ToProto() if err != nil { return err } } else { tree, ok := s.depositTrie.(*trie.SparseMerkleTrie) if !ok { return errors.New("deposit tree was not SparseMerkleTrie") } eth1Data.Trie = tree.ToProto() } return s.cfg.beaconDB.SaveExecutionChainData(ctx, eth1Data) }