package rpc import ( "context" "time" "github.com/golang/protobuf/ptypes/empty" middleware "github.com/grpc-ecosystem/go-grpc-middleware" grpcretry "github.com/grpc-ecosystem/go-grpc-middleware/retry" grpcopentracing "github.com/grpc-ecosystem/go-grpc-middleware/tracing/opentracing" grpcprometheus "github.com/grpc-ecosystem/go-grpc-prometheus" "github.com/pkg/errors" grpcutil "github.com/prysmaticlabs/prysm/api/grpc" ethpb "github.com/prysmaticlabs/prysm/proto/prysm/v1alpha1" validatorpb "github.com/prysmaticlabs/prysm/proto/prysm/v1alpha1/validator-client" "github.com/prysmaticlabs/prysm/validator/client" "google.golang.org/grpc" "google.golang.org/protobuf/types/known/emptypb" ) // Initialize a client connect to a beacon node gRPC endpoint. func (s *Server) registerBeaconClient() error { streamInterceptor := grpc.WithStreamInterceptor(middleware.ChainStreamClient( grpcopentracing.StreamClientInterceptor(), grpcprometheus.StreamClientInterceptor, grpcretry.StreamClientInterceptor(), )) dialOpts := client.ConstructDialOptions( s.clientMaxCallRecvMsgSize, s.clientWithCert, s.clientGrpcRetries, s.clientGrpcRetryDelay, streamInterceptor, ) if dialOpts == nil { return errors.New("no dial options for beacon chain gRPC client") } s.ctx = grpcutil.AppendHeaders(s.ctx, s.clientGrpcHeaders) conn, err := grpc.DialContext(s.ctx, s.beaconClientEndpoint, dialOpts...) if err != nil { return errors.Wrapf(err, "could not dial endpoint: %s", s.beaconClientEndpoint) } if s.clientWithCert != "" { log.Info("Established secure gRPC connection") } s.beaconChainClient = ethpb.NewBeaconChainClient(conn) s.beaconNodeClient = ethpb.NewNodeClient(conn) s.beaconNodeHealthClient = ethpb.NewHealthClient(conn) s.beaconNodeValidatorClient = ethpb.NewBeaconNodeValidatorClient(conn) return nil } // GetBeaconStatus retrieves information about the beacon node gRPC connection // and certain chain metadata, such as the genesis time, the chain head, and the // deposit contract address. func (s *Server) GetBeaconStatus(ctx context.Context, _ *empty.Empty) (*validatorpb.BeaconStatusResponse, error) { syncStatus, err := s.beaconNodeClient.GetSyncStatus(ctx, &emptypb.Empty{}) if err != nil { return &validatorpb.BeaconStatusResponse{ BeaconNodeEndpoint: s.nodeGatewayEndpoint, Connected: false, Syncing: false, }, nil } genesis, err := s.beaconNodeClient.GetGenesis(ctx, &emptypb.Empty{}) if err != nil { return nil, err } genesisTime := uint64(time.Unix(genesis.GenesisTime.Seconds, 0).Unix()) address := genesis.DepositContractAddress chainHead, err := s.beaconChainClient.GetChainHead(ctx, &emptypb.Empty{}) if err != nil { return nil, err } return &validatorpb.BeaconStatusResponse{ BeaconNodeEndpoint: s.beaconClientEndpoint, Connected: true, Syncing: syncStatus.Syncing, GenesisTime: genesisTime, DepositContractAddress: address, ChainHead: chainHead, }, nil } // GetValidatorParticipation is a wrapper around the /eth/v1alpha1 endpoint of the same name. func (s *Server) GetValidatorParticipation( ctx context.Context, req *ethpb.GetValidatorParticipationRequest, ) (*ethpb.ValidatorParticipationResponse, error) { return s.beaconChainClient.GetValidatorParticipation(ctx, req) } // GetValidatorPerformance is a wrapper around the /eth/v1alpha1 endpoint of the same name. func (s *Server) GetValidatorPerformance( ctx context.Context, req *ethpb.ValidatorPerformanceRequest, ) (*ethpb.ValidatorPerformanceResponse, error) { return s.beaconChainClient.GetValidatorPerformance(ctx, req) } // GetValidatorBalances is a wrapper around the /eth/v1alpha1 endpoint of the same name. func (s *Server) GetValidatorBalances( ctx context.Context, req *ethpb.ListValidatorBalancesRequest, ) (*ethpb.ValidatorBalances, error) { return s.beaconChainClient.ListValidatorBalances(ctx, req) } // GetValidators is a wrapper around the /eth/v1alpha1 endpoint of the same name. func (s *Server) GetValidators( ctx context.Context, req *ethpb.ListValidatorsRequest, ) (*ethpb.Validators, error) { return s.beaconChainClient.ListValidators(ctx, req) } // GetValidatorQueue is a wrapper around the /eth/v1alpha1 endpoint of the same name. func (s *Server) GetValidatorQueue( ctx context.Context, _ *empty.Empty, ) (*ethpb.ValidatorQueue, error) { return s.beaconChainClient.GetValidatorQueue(ctx, &emptypb.Empty{}) } // GetPeers is a wrapper around the /eth/v1alpha1 endpoint of the same name. func (s *Server) GetPeers( ctx context.Context, _ *empty.Empty, ) (*ethpb.Peers, error) { return s.beaconNodeClient.ListPeers(ctx, &emptypb.Empty{}) }