package evaluators import ( "context" "github.com/pkg/errors" "github.com/prysmaticlabs/prysm/v4/consensus-types/primitives" ethpb "github.com/prysmaticlabs/prysm/v4/proto/prysm/v1alpha1" "github.com/prysmaticlabs/prysm/v4/testing/endtoend/helpers" "github.com/prysmaticlabs/prysm/v4/testing/endtoend/policies" e2etypes "github.com/prysmaticlabs/prysm/v4/testing/endtoend/types" "github.com/prysmaticlabs/prysm/v4/time/slots" "google.golang.org/grpc" "google.golang.org/protobuf/types/known/emptypb" ) // BuilderIsActive checks that the builder is indeed producing the respective payloads var BuilderIsActive = e2etypes.Evaluator{ Name: "builder_is_active_at_epoch_%d", Policy: policies.OnwardsNthEpoch(helpers.BellatrixE2EForkEpoch), Evaluation: builderActive, } func builderActive(_ *e2etypes.EvaluationContext, conns ...*grpc.ClientConn) error { conn := conns[0] client := ethpb.NewNodeClient(conn) beaconClient := ethpb.NewBeaconChainClient(conn) genesis, err := client.GetGenesis(context.Background(), &emptypb.Empty{}) if err != nil { return errors.Wrap(err, "failed to get genesis data") } currSlot := slots.CurrentSlot(uint64(genesis.GenesisTime.AsTime().Unix())) currEpoch := slots.ToEpoch(currSlot) lowestBound := primitives.Epoch(0) if currEpoch >= 1 { lowestBound = currEpoch - 1 } if lowestBound < helpers.BellatrixE2EForkEpoch { lowestBound = helpers.BellatrixE2EForkEpoch } blockCtrs, err := beaconClient.ListBeaconBlocks(context.Background(), ðpb.ListBlocksRequest{QueryFilter: ðpb.ListBlocksRequest_Epoch{Epoch: lowestBound}}) if err != nil { return errors.Wrap(err, "failed to get beacon blocks") } for _, ctr := range blockCtrs.BlockContainers { b, err := syncCompatibleBlockFromCtr(ctr) if err != nil { return errors.Wrapf(err, "block type doesn't exist for block at epoch %d", lowestBound) } if b.IsNil() { return errors.New("nil block provided") } forkStartSlot, err := slots.EpochStart(helpers.BellatrixE2EForkEpoch) if err != nil { return err } if forkStartSlot == b.Block().Slot() || forkStartSlot+1 == b.Block().Slot() { // Skip fork slot and the next one, as we don't send FCUs yet. continue } execPayload, err := b.Block().Body().Execution() if err != nil { return err } if string(execPayload.ExtraData()) != "prysm-builder" { return errors.Errorf("block with slot %d was not built by the builder. It has an extra data of %s", b.Block().Slot(), string(execPayload.ExtraData())) } } if lowestBound == currEpoch { return nil } blockCtrs, err = beaconClient.ListBeaconBlocks(context.Background(), ðpb.ListBlocksRequest{QueryFilter: ðpb.ListBlocksRequest_Epoch{Epoch: currEpoch}}) if err != nil { return errors.Wrap(err, "failed to get validator participation") } for _, ctr := range blockCtrs.BlockContainers { b, err := syncCompatibleBlockFromCtr(ctr) if err != nil { return errors.Wrapf(err, "block type doesn't exist for block at epoch %d", lowestBound) } if b.IsNil() { return errors.New("nil block provided") } forkStartSlot, err := slots.EpochStart(helpers.BellatrixE2EForkEpoch) if err != nil { return err } if forkStartSlot == b.Block().Slot() || forkStartSlot+1 == b.Block().Slot() { // Skip fork slot and the next one, as we don't send FCUs yet. continue } execPayload, err := b.Block().Body().Execution() if err != nil { return err } if string(execPayload.ExtraData()) != "prysm-builder" { return errors.Errorf("block with slot %d was not built by the builder. It has an extra data of %s", b.Block().Slot(), string(execPayload.ExtraData())) } } return nil }