2019-11-22 05:11:38 +00:00
|
|
|
package aggregator
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
|
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/blockchain"
|
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/core/helpers"
|
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/core/state"
|
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/db"
|
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/sync"
|
|
|
|
pb "github.com/prysmaticlabs/prysm/proto/beacon/rpc/v1"
|
|
|
|
"github.com/prysmaticlabs/prysm/shared/bytesutil"
|
|
|
|
"github.com/sirupsen/logrus"
|
|
|
|
"go.opencensus.io/trace"
|
|
|
|
"google.golang.org/grpc/codes"
|
|
|
|
"google.golang.org/grpc/status"
|
|
|
|
)
|
|
|
|
|
|
|
|
var log logrus.FieldLogger
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
log = logrus.WithField("prefix", "rpc/aggregator")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Server defines a server implementation of the gRPC aggregator service.
|
|
|
|
type Server struct {
|
|
|
|
BeaconDB db.Database
|
|
|
|
HeadFetcher blockchain.HeadFetcher
|
|
|
|
SyncChecker sync.Checker
|
|
|
|
}
|
|
|
|
|
|
|
|
// SubmitAggregateAndProof is called by a validator when its assigned to be an aggregator.
|
|
|
|
// The beacon node will broadcast aggregated attestation and proof on the aggregator's behavior.
|
|
|
|
func (as *Server) SubmitAggregateAndProof(ctx context.Context, req *pb.AggregationRequest) (*pb.AggregationResponse, error) {
|
|
|
|
ctx, span := trace.StartSpan(ctx, "AggregatorServer.SubmitAggregation")
|
|
|
|
defer span.End()
|
|
|
|
span.AddAttributes(trace.Int64Attribute("slot", int64(req.Slot)))
|
|
|
|
|
|
|
|
if as.SyncChecker.Syncing() {
|
|
|
|
return nil, status.Errorf(codes.Unavailable, "Syncing to latest head, not ready to respond")
|
|
|
|
}
|
|
|
|
|
|
|
|
headState, err := as.HeadFetcher.HeadState(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return nil, status.Errorf(codes.Internal, "Could not retrieve head state: %v", err)
|
|
|
|
}
|
2019-12-16 17:43:33 +00:00
|
|
|
|
|
|
|
// Advance slots if it is behind the epoch start of requested slot.
|
|
|
|
// Ex: head slot is 100, req slot is 150, epoch start of 150 is 128. Advance 100 to 128.
|
|
|
|
reqEpochStartSlot := helpers.StartSlot(helpers.SlotToEpoch(req.Slot))
|
|
|
|
if reqEpochStartSlot > headState.Slot {
|
|
|
|
headState, err = state.ProcessSlots(ctx, headState, reqEpochStartSlot)
|
|
|
|
if err != nil {
|
|
|
|
return nil, status.Errorf(codes.Internal, "Could not process slots up to %d: %v", req.Slot, err)
|
|
|
|
}
|
2019-11-22 05:11:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
validatorIndex, exists, err := as.BeaconDB.ValidatorIndex(ctx, bytesutil.ToBytes48(req.PublicKey))
|
|
|
|
if err != nil {
|
|
|
|
return nil, status.Errorf(codes.Internal, "Could not get validator index from DB: %v", err)
|
|
|
|
}
|
|
|
|
if !exists {
|
|
|
|
return nil, status.Error(codes.Internal, "Could not locate validator index in DB")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check if the validator is an aggregator
|
2019-12-16 19:01:40 +00:00
|
|
|
isAggregator, err := helpers.IsAggregator(headState, req.Slot, req.CommitteeIndex, req.SlotSignature)
|
2019-11-22 05:11:38 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, status.Errorf(codes.Internal, "Could not get aggregator status: %v", err)
|
|
|
|
}
|
|
|
|
if !isAggregator {
|
|
|
|
return nil, status.Errorf(codes.InvalidArgument, "Validator is not an aggregator")
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO(3865): Broadcast aggregated attestation & proof via the aggregation topic
|
|
|
|
|
|
|
|
log.WithFields(logrus.Fields{
|
|
|
|
"slot": req.Slot,
|
|
|
|
"validatorIndex": validatorIndex,
|
|
|
|
"committeeIndex": req.CommitteeIndex,
|
2019-11-26 20:36:18 +00:00
|
|
|
}).Debug("Broadcasting aggregated attestation and proof")
|
2019-11-22 05:11:38 +00:00
|
|
|
|
|
|
|
return &pb.AggregationResponse{}, nil
|
|
|
|
}
|