package transition import ( "encoding/binary" "fmt" libcommon "github.com/ledgerwatch/erigon-lib/common" "github.com/Giulio2002/bls" "github.com/ledgerwatch/erigon/cl/clparams" "github.com/ledgerwatch/erigon/cl/cltypes" "github.com/ledgerwatch/erigon/cl/utils" ) func computeSigningRootEpoch(epoch uint64, domain []byte) (libcommon.Hash, error) { b := make([]byte, 32) binary.LittleEndian.PutUint64(b, epoch) hash := utils.Keccak256(b) return utils.Keccak256(hash[:], domain), nil } func (s *StateTransistor) ProcessBlockHeader(block *cltypes.BeaconBlock) error { if block.Slot != s.state.Slot() { return fmt.Errorf("state slot: %d, not equal to block slot: %d", s.state.Slot(), block.Slot) } if block.Slot <= s.state.LatestBlockHeader().Slot { return fmt.Errorf("slock slot: %d, not greater than latest block slot: %d", block.Slot, s.state.LatestBlockHeader().Slot) } propInd, err := s.state.GetBeaconProposerIndex() if err != nil { return fmt.Errorf("error in GetBeaconProposerIndex: %v", err) } if block.ProposerIndex != propInd { return fmt.Errorf("block proposer index: %d, does not match beacon proposer index: %d", block.ProposerIndex, propInd) } latestRoot, err := s.state.LatestBlockHeader().HashSSZ() if err != nil { return fmt.Errorf("unable to hash tree root of latest block header: %v", err) } if block.ParentRoot != latestRoot { return fmt.Errorf("block parent root: %x, does not match latest block root: %x", block.ParentRoot, latestRoot) } bodyRoot, err := block.Body.HashSSZ() if err != nil { return fmt.Errorf("unable to hash tree root of block body: %v", err) } s.state.SetLatestBlockHeader(&cltypes.BeaconBlockHeader{ Slot: block.Slot, ProposerIndex: block.ProposerIndex, ParentRoot: block.ParentRoot, BodyRoot: bodyRoot, }) proposer := s.state.ValidatorAt(int(block.ProposerIndex)) if proposer.Slashed { return fmt.Errorf("proposer: %d is slashed", block.ProposerIndex) } return nil } func (s *StateTransistor) ProcessRandao(randao [96]byte) error { epoch := s.state.Epoch() propInd, err := s.state.GetBeaconProposerIndex() if err != nil { return fmt.Errorf("unable to get proposer index: %v", err) } proposer := s.state.ValidatorAt(int(propInd)) domain, err := s.state.GetDomain(clparams.MainnetBeaconConfig.DomainRandao, epoch) if err != nil { return fmt.Errorf("unable to get domain: %v", err) } signingRoot, err := computeSigningRootEpoch(epoch, domain) if err != nil { return fmt.Errorf("unable to compute signing root: %v", err) } valid, err := bls.Verify(randao[:], signingRoot[:], proposer.PublicKey[:]) if err != nil { return fmt.Errorf("unable to verify public key: %x, with signing root: %x, and signature: %x, %v", proposer.PublicKey[:], signingRoot[:], randao[:], err) } if !valid { return fmt.Errorf("invalid signature: public key: %x, signing root: %x, signature: %x", proposer.PublicKey[:], signingRoot[:], randao[:]) } randaoMixes := s.state.GetRandaoMixes(epoch) randaoHash := utils.Keccak256(randao[:]) mix := [32]byte{} for i := range mix { mix[i] = randaoMixes[i] ^ randaoHash[i] } s.state.SetRandaoMixAt(int(epoch%s.beaconConfig.EpochsPerHistoricalVector), mix) return nil } func (s *StateTransistor) ProcessEth1Data(eth1Data *cltypes.Eth1Data) error { s.state.AddEth1DataVote(eth1Data) newVotes := s.state.Eth1DataVotes() ethDataHash, err := eth1Data.HashSSZ() if err != nil { return fmt.Errorf("unable to get hash tree root of eth1data: %v", err) } // Count how many times body.Eth1Data appears in the votes by comparing their hashes. numVotes := 0 for i := 0; i < len(newVotes); i++ { candidateHash, err := newVotes[i].HashSSZ() if err != nil { return fmt.Errorf("unable to get hash tree root of eth1data: %v", err) } // Check if hash bytes are equal. match := true for i := 0; i < len(candidateHash); i++ { if candidateHash[i] != ethDataHash[i] { match = false } } if match { numVotes += 1 } } if uint64(numVotes*2) > s.beaconConfig.EpochsPerEth1VotingPeriod*s.beaconConfig.SlotsPerEpoch { s.state.SetEth1Data(eth1Data) } return nil }