package validator import ( "bytes" "context" "fmt" "sort" "strconv" "time" "github.com/ethereum/go-ethereum/common" "github.com/ethereum/go-ethereum/common/hexutil" "github.com/golang/protobuf/ptypes/empty" "github.com/pkg/errors" "github.com/prysmaticlabs/prysm/v3/beacon-chain/builder" "github.com/prysmaticlabs/prysm/v3/beacon-chain/cache" "github.com/prysmaticlabs/prysm/v3/beacon-chain/core/helpers" "github.com/prysmaticlabs/prysm/v3/beacon-chain/db/kv" rpchelpers "github.com/prysmaticlabs/prysm/v3/beacon-chain/rpc/eth/helpers" "github.com/prysmaticlabs/prysm/v3/beacon-chain/state" state_native "github.com/prysmaticlabs/prysm/v3/beacon-chain/state/state-native" fieldparams "github.com/prysmaticlabs/prysm/v3/config/fieldparams" "github.com/prysmaticlabs/prysm/v3/config/params" "github.com/prysmaticlabs/prysm/v3/consensus-types/primitives" "github.com/prysmaticlabs/prysm/v3/encoding/bytesutil" ethpbv1 "github.com/prysmaticlabs/prysm/v3/proto/eth/v1" ethpbv2 "github.com/prysmaticlabs/prysm/v3/proto/eth/v2" "github.com/prysmaticlabs/prysm/v3/proto/migration" ethpbalpha "github.com/prysmaticlabs/prysm/v3/proto/prysm/v1alpha1" "github.com/prysmaticlabs/prysm/v3/time/slots" log "github.com/sirupsen/logrus" "go.opencensus.io/trace" "google.golang.org/grpc/codes" "google.golang.org/grpc/status" "google.golang.org/protobuf/types/known/emptypb" ) var errInvalidValIndex = errors.New("invalid validator index") var errParticipation = status.Error(codes.Internal, "Could not obtain epoch participation") // GetAttesterDuties requests the beacon node to provide a set of attestation duties, // which should be performed by validators, for a particular epoch. func (vs *Server) GetAttesterDuties(ctx context.Context, req *ethpbv1.AttesterDutiesRequest) (*ethpbv1.AttesterDutiesResponse, error) { ctx, span := trace.StartSpan(ctx, "validator.GetAttesterDuties") defer span.End() if err := rpchelpers.ValidateSync(ctx, vs.SyncChecker, vs.HeadFetcher, vs.TimeFetcher, vs.OptimisticModeFetcher); err != nil { // We simply return the error because it's already a gRPC error. return nil, err } cs := vs.TimeFetcher.CurrentSlot() currentEpoch := slots.ToEpoch(cs) if req.Epoch > currentEpoch+1 { return nil, status.Errorf(codes.InvalidArgument, "Request epoch %d can not be greater than next epoch %d", req.Epoch, currentEpoch+1) } isOptimistic, err := vs.OptimisticModeFetcher.IsOptimistic(ctx) if err != nil { return nil, status.Errorf(codes.Internal, "Could not check optimistic status: %v", err) } var startSlot primitives.Slot if req.Epoch == currentEpoch+1 { startSlot, err = slots.EpochStart(currentEpoch) } else { startSlot, err = slots.EpochStart(req.Epoch) } if err != nil { return nil, status.Errorf(codes.Internal, "Could not get start slot from epoch %d: %v", req.Epoch, err) } s, err := vs.StateFetcher.StateBySlot(ctx, startSlot) if err != nil { return nil, status.Errorf(codes.Internal, "Could not get state: %v", err) } committeeAssignments, _, err := helpers.CommitteeAssignments(ctx, s, req.Epoch) if err != nil { return nil, status.Errorf(codes.Internal, "Could not compute committee assignments: %v", err) } activeValidatorCount, err := helpers.ActiveValidatorCount(ctx, s, req.Epoch) if err != nil { return nil, status.Errorf(codes.Internal, "Could not get active validator count: %v", err) } committeesAtSlot := helpers.SlotCommitteeCount(activeValidatorCount) duties := make([]*ethpbv1.AttesterDuty, 0, len(req.Index)) for _, index := range req.Index { pubkey := s.PubkeyAtIndex(index) var zeroPubkey [fieldparams.BLSPubkeyLength]byte if bytes.Equal(pubkey[:], zeroPubkey[:]) { return nil, status.Errorf(codes.InvalidArgument, "Invalid validator index") } committee := committeeAssignments[index] if committee == nil { continue } var valIndexInCommittee primitives.CommitteeIndex // valIndexInCommittee will be 0 in case we don't get a match. This is a potential false positive, // however it's an impossible condition because every validator must be assigned to a committee. for cIndex, vIndex := range committee.Committee { if vIndex == index { valIndexInCommittee = primitives.CommitteeIndex(uint64(cIndex)) break } } duties = append(duties, ðpbv1.AttesterDuty{ Pubkey: pubkey[:], ValidatorIndex: index, CommitteeIndex: committee.CommitteeIndex, CommitteeLength: uint64(len(committee.Committee)), CommitteesAtSlot: committeesAtSlot, ValidatorCommitteeIndex: valIndexInCommittee, Slot: committee.AttesterSlot, }) } root, err := attestationDependentRoot(s, req.Epoch) if err != nil { return nil, status.Errorf(codes.Internal, "Could not get dependent root: %v", err) } return ðpbv1.AttesterDutiesResponse{ DependentRoot: root, Data: duties, ExecutionOptimistic: isOptimistic, }, nil } // GetProposerDuties requests beacon node to provide all validators that are scheduled to propose a block in the given epoch. func (vs *Server) GetProposerDuties(ctx context.Context, req *ethpbv1.ProposerDutiesRequest) (*ethpbv1.ProposerDutiesResponse, error) { ctx, span := trace.StartSpan(ctx, "validator.GetProposerDuties") defer span.End() if err := rpchelpers.ValidateSync(ctx, vs.SyncChecker, vs.HeadFetcher, vs.TimeFetcher, vs.OptimisticModeFetcher); err != nil { // We simply return the error because it's already a gRPC error. return nil, err } isOptimistic, err := vs.OptimisticModeFetcher.IsOptimistic(ctx) if err != nil { return nil, status.Errorf(codes.Internal, "Could not check optimistic status: %v", err) } cs := vs.TimeFetcher.CurrentSlot() currentEpoch := slots.ToEpoch(cs) nextEpoch := currentEpoch + 1 var nextEpochLookahead bool if req.Epoch > nextEpoch { return nil, status.Errorf(codes.InvalidArgument, "Request epoch %d can not be greater than next epoch %d", req.Epoch, currentEpoch+1) } else if req.Epoch == nextEpoch { // If the request is for the next epoch, we use the current epoch's state to compute duties. req.Epoch = currentEpoch nextEpochLookahead = true } startSlot, err := slots.EpochStart(req.Epoch) if err != nil { return nil, status.Errorf(codes.Internal, "Could not get start slot from epoch %d: %v", req.Epoch, err) } s, err := vs.StateFetcher.StateBySlot(ctx, startSlot) if err != nil { return nil, status.Errorf(codes.Internal, "Could not get state: %v", err) } var proposals map[primitives.ValidatorIndex][]primitives.Slot if nextEpochLookahead { _, proposals, err = helpers.CommitteeAssignments(ctx, s, nextEpoch) } else { _, proposals, err = helpers.CommitteeAssignments(ctx, s, req.Epoch) } if err != nil { return nil, status.Errorf(codes.Internal, "Could not compute committee assignments: %v", err) } duties := make([]*ethpbv1.ProposerDuty, 0) for index, ss := range proposals { val, err := s.ValidatorAtIndexReadOnly(index) if err != nil { return nil, status.Errorf(codes.Internal, "Could not get validator: %v", err) } pubkey48 := val.PublicKey() pubkey := pubkey48[:] for _, s := range ss { vs.ProposerSlotIndexCache.SetProposerAndPayloadIDs(s, index, [8]byte{} /* payloadID */, [32]byte{} /* head root */) duties = append(duties, ðpbv1.ProposerDuty{ Pubkey: pubkey, ValidatorIndex: index, Slot: s, }) } } sort.Slice(duties, func(i, j int) bool { return duties[i].Slot < duties[j].Slot }) root, err := proposalDependentRoot(s, req.Epoch) if err != nil { return nil, status.Errorf(codes.Internal, "Could not get dependent root: %v", err) } vs.ProposerSlotIndexCache.PrunePayloadIDs(startSlot) return ðpbv1.ProposerDutiesResponse{ DependentRoot: root, Data: duties, ExecutionOptimistic: isOptimistic, }, nil } // GetSyncCommitteeDuties provides a set of sync committee duties for a particular epoch. // // The logic for calculating epoch validity comes from https://ethereum.github.io/beacon-APIs/?urls.primaryName=dev#/Validator/getSyncCommitteeDuties // where `epoch` is described as `epoch // EPOCHS_PER_SYNC_COMMITTEE_PERIOD <= current_epoch // EPOCHS_PER_SYNC_COMMITTEE_PERIOD + 1`. // // Algorithm: // - Get the last valid epoch. This is the last epoch of the next sync committee period. // - Get the state for the requested epoch. If it's a future epoch from the current sync committee period // or an epoch from the next sync committee period, then get the current state. // - Get the state's current sync committee. If it's an epoch from the next sync committee period, then get the next sync committee. // - Get duties. func (vs *Server) GetSyncCommitteeDuties(ctx context.Context, req *ethpbv2.SyncCommitteeDutiesRequest) (*ethpbv2.SyncCommitteeDutiesResponse, error) { ctx, span := trace.StartSpan(ctx, "validator.GetSyncCommitteeDuties") defer span.End() if err := rpchelpers.ValidateSync(ctx, vs.SyncChecker, vs.HeadFetcher, vs.TimeFetcher, vs.OptimisticModeFetcher); err != nil { // We simply return the error because it's already a gRPC error. return nil, err } currentEpoch := slots.ToEpoch(vs.TimeFetcher.CurrentSlot()) lastValidEpoch := syncCommitteeDutiesLastValidEpoch(currentEpoch) if req.Epoch > lastValidEpoch { return nil, status.Errorf(codes.InvalidArgument, "Epoch is too far in the future. Maximum valid epoch is %v.", lastValidEpoch) } requestedEpoch := req.Epoch if requestedEpoch > currentEpoch { requestedEpoch = currentEpoch } slot, err := slots.EpochStart(requestedEpoch) if err != nil { return nil, status.Errorf(codes.Internal, "Could not get sync committee slot: %v", err) } st, err := vs.StateFetcher.State(ctx, []byte(strconv.FormatUint(uint64(slot), 10))) if err != nil { return nil, status.Errorf(codes.Internal, "Could not get sync committee state: %v", err) } currentSyncCommitteeFirstEpoch, err := slots.SyncCommitteePeriodStartEpoch(requestedEpoch) if err != nil { return nil, status.Errorf(codes.InvalidArgument, "Could not get sync committee period start epoch: %v.", err) } nextSyncCommitteeFirstEpoch := currentSyncCommitteeFirstEpoch + params.BeaconConfig().EpochsPerSyncCommitteePeriod var committee *ethpbalpha.SyncCommittee if req.Epoch >= nextSyncCommitteeFirstEpoch { committee, err = st.NextSyncCommittee() if err != nil { return nil, status.Errorf(codes.Internal, "Could not get sync committee: %v", err) } } else { committee, err = st.CurrentSyncCommittee() if err != nil { return nil, status.Errorf(codes.Internal, "Could not get sync committee: %v", err) } } committeePubkeys := make(map[[fieldparams.BLSPubkeyLength]byte][]uint64) for j, pubkey := range committee.Pubkeys { pubkey48 := bytesutil.ToBytes48(pubkey) committeePubkeys[pubkey48] = append(committeePubkeys[pubkey48], uint64(j)) } duties, err := syncCommitteeDuties(req.Index, st, committeePubkeys) if errors.Is(err, errInvalidValIndex) { return nil, status.Error(codes.InvalidArgument, "Invalid validator index") } else if err != nil { return nil, status.Errorf(codes.Internal, "Could not get duties: %v", err) } isOptimistic, err := rpchelpers.IsOptimistic(ctx, st, vs.OptimisticModeFetcher) if err != nil { return nil, status.Errorf(codes.Internal, "Could not check if slot's block is optimistic: %v", err) } return ðpbv2.SyncCommitteeDutiesResponse{ Data: duties, ExecutionOptimistic: isOptimistic, }, nil } // ProduceBlockV2 requests the beacon node to produce a valid unsigned beacon block, which can then be signed by a proposer and submitted. // By definition `/eth/v2/validator/blocks/{slot}`, does not produce block using mev-boost and relayer network. // The following endpoint states that the returned object is a BeaconBlock, not a BlindedBeaconBlock. As such, the block must return a full ExecutionPayload: // https://ethereum.github.io/beacon-APIs/?urls.primaryName=v2.3.0#/Validator/produceBlockV2 // // To use mev-boost and relayer network. It's recommended to use the following endpoint: // https://github.com/ethereum/beacon-APIs/blob/master/apis/validator/blinded_block.yaml func (vs *Server) ProduceBlockV2(ctx context.Context, req *ethpbv1.ProduceBlockRequest) (*ethpbv2.ProduceBlockResponseV2, error) { ctx, span := trace.StartSpan(ctx, "validator.ProduceBlockV2") defer span.End() if err := rpchelpers.ValidateSync(ctx, vs.SyncChecker, vs.HeadFetcher, vs.TimeFetcher, vs.OptimisticModeFetcher); err != nil { // We simply return the error because it's already a gRPC error. return nil, err } v1alpha1req := ðpbalpha.BlockRequest{ Slot: req.Slot, RandaoReveal: req.RandaoReveal, Graffiti: req.Graffiti, SkipMevBoost: true, // Skip mev-boost and relayer network } v1alpha1resp, err := vs.V1Alpha1Server.GetBeaconBlock(ctx, v1alpha1req) if err != nil { // We simply return err because it's already of a gRPC error type. return nil, err } phase0Block, ok := v1alpha1resp.Block.(*ethpbalpha.GenericBeaconBlock_Phase0) if ok { block, err := migration.V1Alpha1ToV1Block(phase0Block.Phase0) if err != nil { return nil, status.Errorf(codes.Internal, "Could not prepare beacon block: %v", err) } return ðpbv2.ProduceBlockResponseV2{ Version: ethpbv2.Version_PHASE0, Data: ðpbv2.BeaconBlockContainerV2{ Block: ðpbv2.BeaconBlockContainerV2_Phase0Block{Phase0Block: block}, }, }, nil } altairBlock, ok := v1alpha1resp.Block.(*ethpbalpha.GenericBeaconBlock_Altair) if ok { block, err := migration.V1Alpha1BeaconBlockAltairToV2(altairBlock.Altair) if err != nil { return nil, status.Errorf(codes.Internal, "Could not prepare beacon block: %v", err) } return ðpbv2.ProduceBlockResponseV2{ Version: ethpbv2.Version_ALTAIR, Data: ðpbv2.BeaconBlockContainerV2{ Block: ðpbv2.BeaconBlockContainerV2_AltairBlock{AltairBlock: block}, }, }, nil } bellatrixBlock, ok := v1alpha1resp.Block.(*ethpbalpha.GenericBeaconBlock_Bellatrix) if ok { block, err := migration.V1Alpha1BeaconBlockBellatrixToV2(bellatrixBlock.Bellatrix) if err != nil { return nil, status.Errorf(codes.Internal, "Could not prepare beacon block: %v", err) } return ðpbv2.ProduceBlockResponseV2{ Version: ethpbv2.Version_BELLATRIX, Data: ðpbv2.BeaconBlockContainerV2{ Block: ðpbv2.BeaconBlockContainerV2_BellatrixBlock{BellatrixBlock: block}, }, }, nil } capellaBlock, ok := v1alpha1resp.Block.(*ethpbalpha.GenericBeaconBlock_Capella) if ok { block, err := migration.V1Alpha1BeaconBlockCapellaToV2(capellaBlock.Capella) if err != nil { return nil, status.Errorf(codes.Internal, "Could not prepare beacon block: %v", err) } return ðpbv2.ProduceBlockResponseV2{ Version: ethpbv2.Version_CAPELLA, Data: ðpbv2.BeaconBlockContainerV2{ Block: ðpbv2.BeaconBlockContainerV2_CapellaBlock{CapellaBlock: block}, }, }, nil } return nil, status.Error(codes.InvalidArgument, "Unsupported block type") } // ProduceBlockV2SSZ requests the beacon node to produce a valid unsigned beacon block, which can then be signed by a proposer and submitted. // // The produced block is in SSZ form. // By definition `/eth/v2/validator/blocks/{slot}/ssz`, does not produce block using mev-boost and relayer network: // The following endpoint states that the returned object is a BeaconBlock, not a BlindedBeaconBlock. As such, the block must return a full ExecutionPayload: // https://ethereum.github.io/beacon-APIs/?urls.primaryName=v2.3.0#/Validator/produceBlockV2 // // To use mev-boost and relayer network. It's recommended to use the following endpoint: // https://github.com/ethereum/beacon-APIs/blob/master/apis/validator/blinded_block.yaml func (vs *Server) ProduceBlockV2SSZ(ctx context.Context, req *ethpbv1.ProduceBlockRequest) (*ethpbv2.SSZContainer, error) { ctx, span := trace.StartSpan(ctx, "validator.ProduceBlockV2SSZ") defer span.End() if err := rpchelpers.ValidateSync(ctx, vs.SyncChecker, vs.HeadFetcher, vs.TimeFetcher, vs.OptimisticModeFetcher); err != nil { // We simply return the error because it's already a gRPC error. return nil, err } v1alpha1req := ðpbalpha.BlockRequest{ Slot: req.Slot, RandaoReveal: req.RandaoReveal, Graffiti: req.Graffiti, SkipMevBoost: true, // Skip mev-boost and relayer network } v1alpha1resp, err := vs.V1Alpha1Server.GetBeaconBlock(ctx, v1alpha1req) if err != nil { // We simply return err because it's already of a gRPC error type. return nil, err } phase0Block, ok := v1alpha1resp.Block.(*ethpbalpha.GenericBeaconBlock_Phase0) if ok { block, err := migration.V1Alpha1ToV1Block(phase0Block.Phase0) if err != nil { return nil, status.Errorf(codes.Internal, "Could not prepare beacon block: %v", err) } sszBlock, err := block.MarshalSSZ() if err != nil { return nil, status.Errorf(codes.Internal, "Could not marshal block into SSZ format: %v", err) } return ðpbv2.SSZContainer{ Version: ethpbv2.Version_PHASE0, Data: sszBlock, }, nil } altairBlock, ok := v1alpha1resp.Block.(*ethpbalpha.GenericBeaconBlock_Altair) if ok { block, err := migration.V1Alpha1BeaconBlockAltairToV2(altairBlock.Altair) if err != nil { return nil, status.Errorf(codes.Internal, "Could not prepare beacon block: %v", err) } sszBlock, err := block.MarshalSSZ() if err != nil { return nil, status.Errorf(codes.Internal, "Could not marshal block into SSZ format: %v", err) } return ðpbv2.SSZContainer{ Version: ethpbv2.Version_ALTAIR, Data: sszBlock, }, nil } bellatrixBlock, ok := v1alpha1resp.Block.(*ethpbalpha.GenericBeaconBlock_Bellatrix) if ok { block, err := migration.V1Alpha1BeaconBlockBellatrixToV2(bellatrixBlock.Bellatrix) if err != nil { return nil, status.Errorf(codes.Internal, "Could not prepare beacon block: %v", err) } sszBlock, err := block.MarshalSSZ() if err != nil { return nil, status.Errorf(codes.Internal, "Could not marshal block into SSZ format: %v", err) } return ðpbv2.SSZContainer{ Version: ethpbv2.Version_BELLATRIX, Data: sszBlock, }, nil } capellaBlock, ok := v1alpha1resp.Block.(*ethpbalpha.GenericBeaconBlock_Capella) if ok { block, err := migration.V1Alpha1BeaconBlockCapellaToV2(capellaBlock.Capella) if err != nil { return nil, status.Errorf(codes.Internal, "Could not prepare beacon block: %v", err) } sszBlock, err := block.MarshalSSZ() if err != nil { return nil, status.Errorf(codes.Internal, "Could not marshal block into SSZ format: %v", err) } return ðpbv2.SSZContainer{ Version: ethpbv2.Version_CAPELLA, Data: sszBlock, }, nil } return nil, status.Error(codes.InvalidArgument, "Unsupported block type") } // ProduceBlindedBlock requests the beacon node to produce a valid unsigned blinded beacon block, // which can then be signed by a proposer and submitted. // // Under the following conditions, this endpoint will return an error. // - The node is syncing or optimistic mode (after bellatrix). // - The builder is not figured (after bellatrix). // - The relayer circuit breaker is activated (after bellatrix). // - The relayer responded with an error (after bellatrix). func (vs *Server) ProduceBlindedBlock(ctx context.Context, req *ethpbv1.ProduceBlockRequest) (*ethpbv2.ProduceBlindedBlockResponse, error) { ctx, span := trace.StartSpan(ctx, "validator.ProduceBlindedBlock") defer span.End() if err := rpchelpers.ValidateSync(ctx, vs.SyncChecker, vs.HeadFetcher, vs.TimeFetcher, vs.OptimisticModeFetcher); err != nil { // We simply return the error because it's already a gRPC error. return nil, err } v1alpha1req := ðpbalpha.BlockRequest{ Slot: req.Slot, RandaoReveal: req.RandaoReveal, Graffiti: req.Graffiti, } v1alpha1resp, err := vs.V1Alpha1Server.GetBeaconBlock(ctx, v1alpha1req) if err != nil { // We simply return err because it's already of a gRPC error type. return nil, err } // Before Bellatrix, return normal block. if req.Slot < primitives.Slot(params.BeaconConfig().BellatrixForkEpoch)*params.BeaconConfig().SlotsPerEpoch { phase0Block, ok := v1alpha1resp.Block.(*ethpbalpha.GenericBeaconBlock_Phase0) if ok { block, err := migration.V1Alpha1ToV1Block(phase0Block.Phase0) if err != nil { return nil, status.Errorf(codes.Internal, "Could not prepare beacon block: %v", err) } return ðpbv2.ProduceBlindedBlockResponse{ Version: ethpbv2.Version_PHASE0, Data: ðpbv2.BlindedBeaconBlockContainer{ Block: ðpbv2.BlindedBeaconBlockContainer_Phase0Block{Phase0Block: block}, }, }, nil } altairBlock, ok := v1alpha1resp.Block.(*ethpbalpha.GenericBeaconBlock_Altair) if ok { block, err := migration.V1Alpha1BeaconBlockAltairToV2(altairBlock.Altair) if err != nil { return nil, status.Errorf(codes.Internal, "Could not prepare beacon block: %v", err) } return ðpbv2.ProduceBlindedBlockResponse{ Version: ethpbv2.Version_ALTAIR, Data: ðpbv2.BlindedBeaconBlockContainer{ Block: ðpbv2.BlindedBeaconBlockContainer_AltairBlock{AltairBlock: block}, }, }, nil } } optimistic, err := vs.OptimisticModeFetcher.IsOptimistic(ctx) if err != nil { return nil, status.Errorf(codes.Internal, "Could not determine if the node is a optimistic node: %v", err) } if optimistic { return nil, status.Errorf(codes.Unavailable, "The node is currently optimistic and cannot serve validators") } bellatrixBlock, ok := v1alpha1resp.Block.(*ethpbalpha.GenericBeaconBlock_BlindedBellatrix) if ok { blk, err := migration.V1Alpha1BeaconBlockBlindedBellatrixToV2Blinded(bellatrixBlock.BlindedBellatrix) if err != nil { return nil, status.Errorf(codes.Internal, "Could not prepare beacon block: %v", err) } return ðpbv2.ProduceBlindedBlockResponse{ Version: ethpbv2.Version_BELLATRIX, Data: ðpbv2.BlindedBeaconBlockContainer{ Block: ðpbv2.BlindedBeaconBlockContainer_BellatrixBlock{BellatrixBlock: blk}, }, }, nil } capellaBlock, ok := v1alpha1resp.Block.(*ethpbalpha.GenericBeaconBlock_BlindedCapella) if ok { blk, err := migration.V1Alpha1BeaconBlockBlindedCapellaToV2Blinded(capellaBlock.BlindedCapella) if err != nil { return nil, status.Errorf(codes.Internal, "Could not prepare beacon block: %v", err) } return ðpbv2.ProduceBlindedBlockResponse{ Version: ethpbv2.Version_CAPELLA, Data: ðpbv2.BlindedBeaconBlockContainer{ Block: ðpbv2.BlindedBeaconBlockContainer_CapellaBlock{CapellaBlock: blk}, }, }, nil } return nil, status.Error(codes.InvalidArgument, "Unsupported block type") } // ProduceBlindedBlockSSZ requests the beacon node to produce a valid unsigned blinded beacon block, // which can then be signed by a proposer and submitted. // // The produced block is in SSZ form. // // Pre-Bellatrix, this endpoint will return a regular block. func (vs *Server) ProduceBlindedBlockSSZ(ctx context.Context, req *ethpbv1.ProduceBlockRequest) (*ethpbv2.SSZContainer, error) { ctx, span := trace.StartSpan(ctx, "validator.ProduceBlindedBlockSSZ") defer span.End() if err := rpchelpers.ValidateSync(ctx, vs.SyncChecker, vs.HeadFetcher, vs.TimeFetcher, vs.OptimisticModeFetcher); err != nil { // We simply return the error because it's already a gRPC error. return nil, err } v1alpha1req := ðpbalpha.BlockRequest{ Slot: req.Slot, RandaoReveal: req.RandaoReveal, Graffiti: req.Graffiti, } v1alpha1resp, err := vs.V1Alpha1Server.GetBeaconBlock(ctx, v1alpha1req) if err != nil { // We simply return err because it's already of a gRPC error type. return nil, err } phase0Block, ok := v1alpha1resp.Block.(*ethpbalpha.GenericBeaconBlock_Phase0) if ok { block, err := migration.V1Alpha1ToV1Block(phase0Block.Phase0) if err != nil { return nil, status.Errorf(codes.Internal, "Could not prepare beacon block: %v", err) } sszBlock, err := block.MarshalSSZ() if err != nil { return nil, status.Errorf(codes.Internal, "Could not marshal block into SSZ format: %v", err) } return ðpbv2.SSZContainer{ Version: ethpbv2.Version_PHASE0, Data: sszBlock, }, nil } altairBlock, ok := v1alpha1resp.Block.(*ethpbalpha.GenericBeaconBlock_Altair) if ok { block, err := migration.V1Alpha1BeaconBlockAltairToV2(altairBlock.Altair) if err != nil { return nil, status.Errorf(codes.Internal, "Could not prepare beacon block: %v", err) } sszBlock, err := block.MarshalSSZ() if err != nil { return nil, status.Errorf(codes.Internal, "Could not marshal block into SSZ format: %v", err) } return ðpbv2.SSZContainer{ Version: ethpbv2.Version_ALTAIR, Data: sszBlock, }, nil } bellatrixBlock, ok := v1alpha1resp.Block.(*ethpbalpha.GenericBeaconBlock_Bellatrix) if ok { block, err := migration.V1Alpha1BeaconBlockBellatrixToV2Blinded(bellatrixBlock.Bellatrix) if err != nil { return nil, status.Errorf(codes.Internal, "Could not prepare beacon block: %v", err) } sszBlock, err := block.MarshalSSZ() if err != nil { return nil, status.Errorf(codes.Internal, "Could not marshal block into SSZ format: %v", err) } return ðpbv2.SSZContainer{ Version: ethpbv2.Version_BELLATRIX, Data: sszBlock, }, nil } capellaBlock, ok := v1alpha1resp.Block.(*ethpbalpha.GenericBeaconBlock_Capella) if ok { block, err := migration.V1Alpha1BeaconBlockCapellaToV2Blinded(capellaBlock.Capella) if err != nil { return nil, status.Errorf(codes.Internal, "Could not prepare beacon block: %v", err) } sszBlock, err := block.MarshalSSZ() if err != nil { return nil, status.Errorf(codes.Internal, "Could not marshal block into SSZ format: %v", err) } return ðpbv2.SSZContainer{ Version: ethpbv2.Version_CAPELLA, Data: sszBlock, }, nil } return nil, status.Error(codes.InvalidArgument, "Unsupported block type") } // PrepareBeaconProposer caches and updates the fee recipient for the given proposer. func (vs *Server) PrepareBeaconProposer( ctx context.Context, request *ethpbv1.PrepareBeaconProposerRequest, ) (*emptypb.Empty, error) { ctx, span := trace.StartSpan(ctx, "validator.PrepareBeaconProposer") defer span.End() var feeRecipients []common.Address var validatorIndices []primitives.ValidatorIndex newRecipients := make([]*ethpbv1.PrepareBeaconProposerRequest_FeeRecipientContainer, 0, len(request.Recipients)) for _, r := range request.Recipients { f, err := vs.V1Alpha1Server.BeaconDB.FeeRecipientByValidatorID(ctx, r.ValidatorIndex) switch { case errors.Is(err, kv.ErrNotFoundFeeRecipient): newRecipients = append(newRecipients, r) case err != nil: return nil, status.Errorf(codes.Internal, "Could not get fee recipient by validator index: %v", err) default: if common.BytesToAddress(r.FeeRecipient) != f { newRecipients = append(newRecipients, r) } } } if len(newRecipients) == 0 { return &emptypb.Empty{}, nil } for _, recipientContainer := range newRecipients { recipient := hexutil.Encode(recipientContainer.FeeRecipient) if !common.IsHexAddress(recipient) { return nil, status.Errorf(codes.InvalidArgument, fmt.Sprintf("Invalid fee recipient address: %v", recipient)) } feeRecipients = append(feeRecipients, common.BytesToAddress(recipientContainer.FeeRecipient)) validatorIndices = append(validatorIndices, recipientContainer.ValidatorIndex) } if err := vs.V1Alpha1Server.BeaconDB.SaveFeeRecipientsByValidatorIDs(ctx, validatorIndices, feeRecipients); err != nil { return nil, status.Errorf(codes.Internal, "Could not save fee recipients: %v", err) } log.WithFields(log.Fields{ "validatorIndices": validatorIndices, }).Info("Updated fee recipient addresses for validator indices") return &emptypb.Empty{}, nil } // SubmitValidatorRegistration submits validator registrations. func (vs *Server) SubmitValidatorRegistration(ctx context.Context, reg *ethpbv1.SubmitValidatorRegistrationsRequest) (*empty.Empty, error) { ctx, span := trace.StartSpan(ctx, "validator.SubmitValidatorRegistration") defer span.End() if vs.V1Alpha1Server.BlockBuilder == nil || !vs.V1Alpha1Server.BlockBuilder.Configured() { return &empty.Empty{}, status.Errorf(codes.Internal, "Could not register block builder: %v", builder.ErrNoBuilder) } var registrations []*ethpbalpha.SignedValidatorRegistrationV1 for i, registration := range reg.Registrations { message := reg.Registrations[i].Message registrations = append(registrations, ðpbalpha.SignedValidatorRegistrationV1{ Message: ðpbalpha.ValidatorRegistrationV1{ FeeRecipient: message.FeeRecipient, GasLimit: message.GasLimit, Timestamp: message.Timestamp, Pubkey: message.Pubkey, }, Signature: registration.Signature, }) } if len(registrations) == 0 { return &empty.Empty{}, status.Errorf(codes.InvalidArgument, "Validator registration request is empty") } if err := vs.V1Alpha1Server.BlockBuilder.RegisterValidator(ctx, registrations); err != nil { return nil, status.Errorf(codes.InvalidArgument, "Could not register block builder: %v", err) } return &empty.Empty{}, nil } // ProduceAttestationData requests that the beacon node produces attestation data for // the requested committee index and slot based on the nodes current head. func (vs *Server) ProduceAttestationData(ctx context.Context, req *ethpbv1.ProduceAttestationDataRequest) (*ethpbv1.ProduceAttestationDataResponse, error) { ctx, span := trace.StartSpan(ctx, "validator.ProduceAttestationData") defer span.End() v1alpha1req := ðpbalpha.AttestationDataRequest{ Slot: req.Slot, CommitteeIndex: req.CommitteeIndex, } v1alpha1resp, err := vs.V1Alpha1Server.GetAttestationData(ctx, v1alpha1req) if err != nil { // We simply return err because it's already of a gRPC error type. return nil, err } attData := migration.V1Alpha1AttDataToV1(v1alpha1resp) return ðpbv1.ProduceAttestationDataResponse{Data: attData}, nil } // GetAggregateAttestation aggregates all attestations matching the given attestation data root and slot, returning the aggregated result. func (vs *Server) GetAggregateAttestation(ctx context.Context, req *ethpbv1.AggregateAttestationRequest) (*ethpbv1.AggregateAttestationResponse, error) { _, span := trace.StartSpan(ctx, "validator.GetAggregateAttestation") defer span.End() if err := vs.AttestationsPool.AggregateUnaggregatedAttestations(ctx); err != nil { return nil, status.Errorf(codes.Internal, "Could not aggregate unaggregated attestations: %v", err) } allAtts := vs.AttestationsPool.AggregatedAttestations() var bestMatchingAtt *ethpbalpha.Attestation for _, att := range allAtts { if att.Data.Slot == req.Slot { root, err := att.Data.HashTreeRoot() if err != nil { return nil, status.Errorf(codes.Internal, "Could not get attestation data root: %v", err) } if bytes.Equal(root[:], req.AttestationDataRoot) { if bestMatchingAtt == nil || len(att.AggregationBits) > len(bestMatchingAtt.AggregationBits) { bestMatchingAtt = att } } } } if bestMatchingAtt == nil { return nil, status.Error(codes.NotFound, "No matching attestation found") } return ðpbv1.AggregateAttestationResponse{ Data: migration.V1Alpha1AttestationToV1(bestMatchingAtt), }, nil } // SubmitAggregateAndProofs verifies given aggregate and proofs and publishes them on appropriate gossipsub topic. func (vs *Server) SubmitAggregateAndProofs(ctx context.Context, req *ethpbv1.SubmitAggregateAndProofsRequest) (*empty.Empty, error) { ctx, span := trace.StartSpan(ctx, "validator.SubmitAggregateAndProofs") defer span.End() for _, agg := range req.Data { if agg == nil || agg.Message == nil || agg.Message.Aggregate == nil || agg.Message.Aggregate.Data == nil { return nil, status.Error(codes.InvalidArgument, "Signed aggregate request can't be nil") } sigLen := fieldparams.BLSSignatureLength emptySig := make([]byte, sigLen) if bytes.Equal(agg.Signature, emptySig) || bytes.Equal(agg.Message.SelectionProof, emptySig) || bytes.Equal(agg.Message.Aggregate.Signature, emptySig) { return nil, status.Error(codes.InvalidArgument, "Signed signatures can't be zero hashes") } if len(agg.Signature) != sigLen || len(agg.Message.Aggregate.Signature) != sigLen { return nil, status.Errorf(codes.InvalidArgument, "Incorrect signature length. Expected %d bytes", sigLen) } // As a preventive measure, a beacon node shouldn't broadcast an attestation whose slot is out of range. if err := helpers.ValidateAttestationTime(agg.Message.Aggregate.Data.Slot, vs.TimeFetcher.GenesisTime(), params.BeaconNetworkConfig().MaximumGossipClockDisparity); err != nil { return nil, status.Error(codes.InvalidArgument, "Attestation slot is no longer valid from current time") } } broadcastFailed := false for _, agg := range req.Data { v1alpha1Agg := migration.V1SignedAggregateAttAndProofToV1Alpha1(agg) if err := vs.Broadcaster.Broadcast(ctx, v1alpha1Agg); err != nil { broadcastFailed = true } else { log.WithFields(log.Fields{ "slot": agg.Message.Aggregate.Data.Slot, "committeeIndex": agg.Message.Aggregate.Data.Index, "validatorIndex": agg.Message.AggregatorIndex, "aggregatedCount": agg.Message.Aggregate.AggregationBits.Count(), }).Debug("Broadcasting aggregated attestation and proof") } } if broadcastFailed { return nil, status.Errorf( codes.Internal, "Could not broadcast one or more signed aggregated attestations") } return &emptypb.Empty{}, nil } // SubmitBeaconCommitteeSubscription searches using discv5 for peers related to the provided subnet information // and replaces current peers with those ones if necessary. func (vs *Server) SubmitBeaconCommitteeSubscription(ctx context.Context, req *ethpbv1.SubmitBeaconCommitteeSubscriptionsRequest) (*emptypb.Empty, error) { ctx, span := trace.StartSpan(ctx, "validator.SubmitBeaconCommitteeSubscription") defer span.End() if err := rpchelpers.ValidateSync(ctx, vs.SyncChecker, vs.HeadFetcher, vs.TimeFetcher, vs.OptimisticModeFetcher); err != nil { // We simply return the error because it's already a gRPC error. return nil, err } if len(req.Data) == 0 { return nil, status.Error(codes.InvalidArgument, "No subscriptions provided") } s, err := vs.HeadFetcher.HeadStateReadOnly(ctx) if err != nil { return nil, status.Errorf(codes.Internal, "Could not get head state: %v", err) } // Verify validators at the beginning to return early if request is invalid. validators := make([]state.ReadOnlyValidator, len(req.Data)) for i, sub := range req.Data { val, err := s.ValidatorAtIndexReadOnly(sub.ValidatorIndex) if outOfRangeErr, ok := err.(*state_native.ValidatorIndexOutOfRangeError); ok { return nil, status.Errorf(codes.InvalidArgument, "Invalid validator ID: %v", outOfRangeErr) } validators[i] = val } fetchValsLen := func(slot primitives.Slot) (uint64, error) { wantedEpoch := slots.ToEpoch(slot) vals, err := vs.HeadFetcher.HeadValidatorsIndices(ctx, wantedEpoch) if err != nil { return 0, err } return uint64(len(vals)), nil } // Request the head validator indices of epoch represented by the first requested slot. currValsLen, err := fetchValsLen(req.Data[0].Slot) if err != nil { return nil, status.Errorf(codes.Internal, "Could not retrieve head validator length: %v", err) } currEpoch := slots.ToEpoch(req.Data[0].Slot) for _, sub := range req.Data { // If epoch has changed, re-request active validators length if currEpoch != slots.ToEpoch(sub.Slot) { currValsLen, err = fetchValsLen(sub.Slot) if err != nil { return nil, status.Errorf(codes.Internal, "Could not retrieve head validator length: %v", err) } currEpoch = slots.ToEpoch(sub.Slot) } subnet := helpers.ComputeSubnetFromCommitteeAndSlot(currValsLen, sub.CommitteeIndex, sub.Slot) cache.SubnetIDs.AddAttesterSubnetID(sub.Slot, subnet) if sub.IsAggregator { cache.SubnetIDs.AddAggregatorSubnetID(sub.Slot, subnet) } } for _, val := range validators { valStatus, err := rpchelpers.ValidatorStatus(val, currEpoch) if err != nil { return nil, status.Errorf(codes.Internal, "Could not retrieve validator status: %v", err) } pubkey := val.PublicKey() vs.V1Alpha1Server.AssignValidatorToSubnet(pubkey[:], v1ValidatorStatusToV1Alpha1(valStatus)) } return &emptypb.Empty{}, nil } // SubmitSyncCommitteeSubscription subscribe to a number of sync committee subnets. // // Subscribing to sync committee subnets is an action performed by VC to enable // network participation in Altair networks, and only required if the VC has an active // validator in an active sync committee. func (vs *Server) SubmitSyncCommitteeSubscription(ctx context.Context, req *ethpbv2.SubmitSyncCommitteeSubscriptionsRequest) (*empty.Empty, error) { ctx, span := trace.StartSpan(ctx, "validator.SubmitSyncCommitteeSubscription") defer span.End() if err := rpchelpers.ValidateSync(ctx, vs.SyncChecker, vs.HeadFetcher, vs.TimeFetcher, vs.OptimisticModeFetcher); err != nil { // We simply return the error because it's already a gRPC error. return nil, err } if len(req.Data) == 0 { return nil, status.Error(codes.InvalidArgument, "No subscriptions provided") } s, err := vs.HeadFetcher.HeadStateReadOnly(ctx) if err != nil { return nil, status.Errorf(codes.Internal, "Could not get head state: %v", err) } currEpoch := slots.ToEpoch(s.Slot()) validators := make([]state.ReadOnlyValidator, len(req.Data)) for i, sub := range req.Data { val, err := s.ValidatorAtIndexReadOnly(sub.ValidatorIndex) if err != nil { return nil, status.Errorf(codes.Internal, "Could not get validator at index %d: %v", sub.ValidatorIndex, err) } valStatus, err := rpchelpers.ValidatorSubStatus(val, currEpoch) if err != nil { return nil, status.Errorf(codes.Internal, "Could not get validator status at index %d: %v", sub.ValidatorIndex, err) } if valStatus != ethpbv1.ValidatorStatus_ACTIVE_ONGOING && valStatus != ethpbv1.ValidatorStatus_ACTIVE_EXITING { return nil, status.Errorf(codes.InvalidArgument, "Validator at index %d is not active or exiting: %v", sub.ValidatorIndex, err) } validators[i] = val } startEpoch, err := slots.SyncCommitteePeriodStartEpoch(currEpoch) if err != nil { return nil, status.Errorf(codes.Internal, "Could not get sync committee period start epoch: %v.", err) } for i, sub := range req.Data { if sub.UntilEpoch <= currEpoch { return nil, status.Errorf(codes.InvalidArgument, "Epoch for subscription at index %d is in the past. It must be at least %d", i, currEpoch+1) } maxValidUntilEpoch := startEpoch + params.BeaconConfig().EpochsPerSyncCommitteePeriod*2 if sub.UntilEpoch > maxValidUntilEpoch { return nil, status.Errorf( codes.InvalidArgument, "Epoch for subscription at index %d is too far in the future. It can be at most %d", i, maxValidUntilEpoch, ) } } for i, sub := range req.Data { pubkey48 := validators[i].PublicKey() // Handle overflow in the event current epoch is less than end epoch. // This is an impossible condition, so it is a defensive check. epochsToWatch, err := sub.UntilEpoch.SafeSub(uint64(startEpoch)) if err != nil { epochsToWatch = 0 } epochDuration := time.Duration(params.BeaconConfig().SlotsPerEpoch.Mul(params.BeaconConfig().SecondsPerSlot)) * time.Second totalDuration := epochDuration * time.Duration(epochsToWatch) cache.SyncSubnetIDs.AddSyncCommitteeSubnets(pubkey48[:], startEpoch, sub.SyncCommitteeIndices, totalDuration) } return &empty.Empty{}, nil } // ProduceSyncCommitteeContribution requests that the beacon node produce a sync committee contribution. func (vs *Server) ProduceSyncCommitteeContribution( ctx context.Context, req *ethpbv2.ProduceSyncCommitteeContributionRequest, ) (*ethpbv2.ProduceSyncCommitteeContributionResponse, error) { ctx, span := trace.StartSpan(ctx, "validator.ProduceSyncCommitteeContribution") defer span.End() msgs, err := vs.SyncCommitteePool.SyncCommitteeMessages(req.Slot) if err != nil { return nil, status.Errorf(codes.Internal, "Could not get sync subcommittee messages: %v", err) } if msgs == nil { return nil, status.Errorf(codes.NotFound, "No subcommittee messages found") } aggregatedSig, bits, err := vs.V1Alpha1Server.AggregatedSigAndAggregationBits(ctx, msgs, req.Slot, req.SubcommitteeIndex, req.BeaconBlockRoot) if err != nil { return nil, status.Errorf(codes.Internal, "Could not get contribution data: %v", err) } contribution := ðpbv2.SyncCommitteeContribution{ Slot: req.Slot, BeaconBlockRoot: req.BeaconBlockRoot, SubcommitteeIndex: req.SubcommitteeIndex, AggregationBits: bits, Signature: aggregatedSig, } return ðpbv2.ProduceSyncCommitteeContributionResponse{ Data: contribution, }, nil } // SubmitContributionAndProofs publishes multiple signed sync committee contribution and proofs. func (vs *Server) SubmitContributionAndProofs(ctx context.Context, req *ethpbv2.SubmitContributionAndProofsRequest) (*empty.Empty, error) { ctx, span := trace.StartSpan(ctx, "validator.SubmitContributionAndProofs") defer span.End() for _, item := range req.Data { v1alpha1Req := ðpbalpha.SignedContributionAndProof{ Message: ðpbalpha.ContributionAndProof{ AggregatorIndex: item.Message.AggregatorIndex, Contribution: ðpbalpha.SyncCommitteeContribution{ Slot: item.Message.Contribution.Slot, BlockRoot: item.Message.Contribution.BeaconBlockRoot, SubcommitteeIndex: item.Message.Contribution.SubcommitteeIndex, AggregationBits: item.Message.Contribution.AggregationBits, Signature: item.Message.Contribution.Signature, }, SelectionProof: item.Message.SelectionProof, }, Signature: item.Signature, } _, err := vs.V1Alpha1Server.SubmitSignedContributionAndProof(ctx, v1alpha1Req) // We simply return err because it's already of a gRPC error type. if err != nil { return nil, err } } return &empty.Empty{}, nil } // GetLiveness requests the beacon node to indicate if a validator has been observed to be live in a given epoch. // The beacon node might detect liveness by observing messages from the validator on the network, // in the beacon chain, from its API or from any other source. // A beacon node SHOULD support the current and previous epoch, however it MAY support earlier epoch. // It is important to note that the values returned by the beacon node are not canonical; // they are best-effort and based upon a subjective view of the network. // A beacon node that was recently started or suffered a network partition may indicate that a validator is not live when it actually is. func (vs *Server) GetLiveness(ctx context.Context, req *ethpbv2.GetLivenessRequest) (*ethpbv2.GetLivenessResponse, error) { ctx, span := trace.StartSpan(ctx, "validator.GetLiveness") defer span.End() var participation []byte // First we check if the requested epoch is the current epoch. // If it is, then we won't be able to fetch the state at the end of the epoch. // In that case we get participation info from the head state. // We can also use the head state to get participation info for the previous epoch. headSt, err := vs.HeadFetcher.HeadState(ctx) if err != nil { return nil, status.Error(codes.Internal, "Could not get head state") } currEpoch := slots.ToEpoch(headSt.Slot()) if req.Epoch > currEpoch { return nil, status.Error(codes.InvalidArgument, "Requested epoch cannot be in the future") } var st state.BeaconState if req.Epoch == currEpoch { st = headSt participation, err = st.CurrentEpochParticipation() if err != nil { return nil, errParticipation } } else if req.Epoch == currEpoch-1 { st = headSt participation, err = st.PreviousEpochParticipation() if err != nil { return nil, errParticipation } } else { epochEnd, err := slots.EpochEnd(req.Epoch) if err != nil { return nil, status.Error(codes.Internal, "Could not get requested epoch's end slot") } st, err = vs.StateFetcher.StateBySlot(ctx, epochEnd) if err != nil { return nil, status.Error(codes.Internal, "Could not get slot for requested epoch") } participation, err = st.CurrentEpochParticipation() if err != nil { return nil, errParticipation } } resp := ðpbv2.GetLivenessResponse{ Data: make([]*ethpbv2.GetLivenessResponse_Liveness, len(req.Index)), } for i, vi := range req.Index { if vi >= primitives.ValidatorIndex(len(participation)) { return nil, status.Errorf(codes.InvalidArgument, "Validator index %d is invalid", vi) } resp.Data[i] = ðpbv2.GetLivenessResponse_Liveness{ Index: vi, IsLive: participation[vi] != 0, } } return resp, nil } // attestationDependentRoot is get_block_root_at_slot(state, compute_start_slot_at_epoch(epoch - 1) - 1) // or the genesis block root in the case of underflow. func attestationDependentRoot(s state.BeaconState, epoch primitives.Epoch) ([]byte, error) { var dependentRootSlot primitives.Slot if epoch <= 1 { dependentRootSlot = 0 } else { prevEpochStartSlot, err := slots.EpochStart(epoch.Sub(1)) if err != nil { return nil, status.Errorf(codes.Internal, "Could not obtain epoch's start slot: %v", err) } dependentRootSlot = prevEpochStartSlot.Sub(1) } root, err := helpers.BlockRootAtSlot(s, dependentRootSlot) if err != nil { return nil, errors.Wrap(err, "could not get block root") } return root, nil } // proposalDependentRoot is get_block_root_at_slot(state, compute_start_slot_at_epoch(epoch) - 1) // or the genesis block root in the case of underflow. func proposalDependentRoot(s state.BeaconState, epoch primitives.Epoch) ([]byte, error) { var dependentRootSlot primitives.Slot if epoch == 0 { dependentRootSlot = 0 } else { epochStartSlot, err := slots.EpochStart(epoch) if err != nil { return nil, status.Errorf(codes.Internal, "Could not obtain epoch's start slot: %v", err) } dependentRootSlot = epochStartSlot.Sub(1) } root, err := helpers.BlockRootAtSlot(s, dependentRootSlot) if err != nil { return nil, errors.Wrap(err, "could not get block root") } return root, nil } // Logic based on https://hackmd.io/ofFJ5gOmQpu1jjHilHbdQQ func v1ValidatorStatusToV1Alpha1(valStatus ethpbv1.ValidatorStatus) ethpbalpha.ValidatorStatus { switch valStatus { case ethpbv1.ValidatorStatus_ACTIVE: return ethpbalpha.ValidatorStatus_ACTIVE case ethpbv1.ValidatorStatus_PENDING: return ethpbalpha.ValidatorStatus_PENDING case ethpbv1.ValidatorStatus_WITHDRAWAL: return ethpbalpha.ValidatorStatus_EXITED default: return ethpbalpha.ValidatorStatus_UNKNOWN_STATUS } } func syncCommitteeDutiesLastValidEpoch(currentEpoch primitives.Epoch) primitives.Epoch { currentSyncPeriodIndex := currentEpoch / params.BeaconConfig().EpochsPerSyncCommitteePeriod // Return the last epoch of the next sync committee. // To do this we go two periods ahead to find the first invalid epoch, and then subtract 1. return (currentSyncPeriodIndex+2)*params.BeaconConfig().EpochsPerSyncCommitteePeriod - 1 } func syncCommitteeDuties( valIndices []primitives.ValidatorIndex, st state.BeaconState, committeePubkeys map[[fieldparams.BLSPubkeyLength]byte][]uint64, ) ([]*ethpbv2.SyncCommitteeDuty, error) { duties := make([]*ethpbv2.SyncCommitteeDuty, 0) for _, index := range valIndices { duty := ðpbv2.SyncCommitteeDuty{ ValidatorIndex: index, } valPubkey48 := st.PubkeyAtIndex(index) var zeroPubkey [fieldparams.BLSPubkeyLength]byte if bytes.Equal(valPubkey48[:], zeroPubkey[:]) { return nil, errInvalidValIndex } valPubkey := valPubkey48[:] duty.Pubkey = valPubkey indices, ok := committeePubkeys[valPubkey48] if ok { duty.ValidatorSyncCommitteeIndices = indices duties = append(duties, duty) } } return duties, nil }