2021-09-01 23:08:05 +00:00
|
|
|
package validator
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"context"
|
|
|
|
|
|
|
|
"github.com/pkg/errors"
|
|
|
|
types "github.com/prysmaticlabs/eth2-types"
|
2021-10-28 08:56:22 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/core/feed"
|
|
|
|
opfeed "github.com/prysmaticlabs/prysm/beacon-chain/core/feed/operation"
|
2021-09-21 19:59:25 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/config/params"
|
2021-09-15 22:55:11 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/crypto/bls"
|
2021-09-23 15:23:37 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/encoding/bytesutil"
|
2021-09-01 23:08:05 +00:00
|
|
|
ethpb "github.com/prysmaticlabs/prysm/proto/prysm/v1alpha1"
|
|
|
|
"golang.org/x/sync/errgroup"
|
|
|
|
"google.golang.org/grpc/codes"
|
|
|
|
"google.golang.org/grpc/status"
|
|
|
|
"google.golang.org/protobuf/types/known/emptypb"
|
|
|
|
)
|
|
|
|
|
|
|
|
// GetSyncMessageBlockRoot retrieves the sync committee block root of the beacon chain.
|
|
|
|
func (vs *Server) GetSyncMessageBlockRoot(
|
|
|
|
ctx context.Context, _ *emptypb.Empty,
|
|
|
|
) (*ethpb.SyncMessageBlockRootResponse, error) {
|
|
|
|
r, err := vs.HeadFetcher.HeadRoot(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return nil, status.Errorf(codes.Internal, "Could not retrieve head root: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return ðpb.SyncMessageBlockRootResponse{
|
|
|
|
Root: r,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// SubmitSyncMessage submits the sync committee message to the network.
|
|
|
|
// It also saves the sync committee message into the pending pool for block inclusion.
|
|
|
|
func (vs *Server) SubmitSyncMessage(ctx context.Context, msg *ethpb.SyncCommitteeMessage) (*emptypb.Empty, error) {
|
|
|
|
errs, ctx := errgroup.WithContext(ctx)
|
|
|
|
|
2021-09-20 18:06:43 +00:00
|
|
|
headSyncCommitteeIndices, err := vs.HeadFetcher.HeadSyncCommitteeIndices(ctx, msg.ValidatorIndex, msg.Slot)
|
2021-09-01 23:08:05 +00:00
|
|
|
if err != nil {
|
|
|
|
return &emptypb.Empty{}, err
|
|
|
|
}
|
|
|
|
// Broadcasting and saving message into the pool in parallel. As one fail should not affect another.
|
|
|
|
// This broadcasts for all subnets.
|
2021-09-20 18:06:43 +00:00
|
|
|
for _, index := range headSyncCommitteeIndices {
|
2021-09-01 23:08:05 +00:00
|
|
|
subCommitteeSize := params.BeaconConfig().SyncCommitteeSize / params.BeaconConfig().SyncCommitteeSubnetCount
|
2021-09-20 18:06:43 +00:00
|
|
|
subnet := uint64(index) / subCommitteeSize
|
2021-09-01 23:08:05 +00:00
|
|
|
errs.Go(func() error {
|
|
|
|
return vs.P2P.BroadcastSyncCommitteeMessage(ctx, subnet, msg)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := vs.SyncCommitteePool.SaveSyncCommitteeMessage(msg); err != nil {
|
|
|
|
return &emptypb.Empty{}, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Wait for p2p broadcast to complete and return the first error (if any)
|
|
|
|
err = errs.Wait()
|
|
|
|
return &emptypb.Empty{}, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetSyncSubcommitteeIndex is called by a sync committee participant to get
|
|
|
|
// its subcommittee index for sync message aggregation duty.
|
|
|
|
func (vs *Server) GetSyncSubcommitteeIndex(
|
|
|
|
ctx context.Context, req *ethpb.SyncSubcommitteeIndexRequest,
|
|
|
|
) (*ethpb.SyncSubcommitteeIndexResponse, error) {
|
2022-01-20 02:37:27 +00:00
|
|
|
index, exists := vs.HeadFetcher.HeadPublicKeyToValidatorIndex(bytesutil.ToBytes48(req.PublicKey))
|
2021-09-01 23:08:05 +00:00
|
|
|
if !exists {
|
|
|
|
return nil, errors.New("public key does not exist in state")
|
|
|
|
}
|
2021-09-20 18:06:43 +00:00
|
|
|
indices, err := vs.HeadFetcher.HeadSyncCommitteeIndices(ctx, index, req.Slot)
|
2021-09-01 23:08:05 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, status.Errorf(codes.Internal, "Could not get sync subcommittee index: %v", err)
|
|
|
|
}
|
2021-09-20 18:06:43 +00:00
|
|
|
return ðpb.SyncSubcommitteeIndexResponse{Indices: indices}, nil
|
2021-09-01 23:08:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// GetSyncCommitteeContribution is called by a sync committee aggregator
|
|
|
|
// to retrieve sync committee contribution object.
|
|
|
|
func (vs *Server) GetSyncCommitteeContribution(
|
|
|
|
ctx context.Context, req *ethpb.SyncCommitteeContributionRequest,
|
|
|
|
) (*ethpb.SyncCommitteeContribution, error) {
|
|
|
|
msgs, err := vs.SyncCommitteePool.SyncCommitteeMessages(req.Slot)
|
|
|
|
if err != nil {
|
|
|
|
return nil, status.Errorf(codes.Internal, "Could not get sync subcommittee messages: %v", err)
|
|
|
|
}
|
|
|
|
headRoot, err := vs.HeadFetcher.HeadRoot(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return nil, status.Errorf(codes.Internal, "Could not get head root: %v", err)
|
|
|
|
}
|
2021-09-03 14:47:35 +00:00
|
|
|
aggregatedSig, bits, err := vs.AggregatedSigAndAggregationBits(ctx, msgs, req.Slot, req.SubnetId, headRoot)
|
|
|
|
if err != nil {
|
|
|
|
return nil, status.Errorf(codes.Internal, "Could not get contribution data: %v", err)
|
2021-09-01 23:08:05 +00:00
|
|
|
}
|
|
|
|
contribution := ðpb.SyncCommitteeContribution{
|
|
|
|
Slot: req.Slot,
|
|
|
|
BlockRoot: headRoot,
|
|
|
|
SubcommitteeIndex: req.SubnetId,
|
|
|
|
AggregationBits: bits,
|
|
|
|
Signature: aggregatedSig,
|
|
|
|
}
|
|
|
|
|
|
|
|
return contribution, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// SubmitSignedContributionAndProof is called by a sync committee aggregator
|
|
|
|
// to submit signed contribution and proof object.
|
|
|
|
func (vs *Server) SubmitSignedContributionAndProof(
|
|
|
|
ctx context.Context, s *ethpb.SignedContributionAndProof,
|
|
|
|
) (*emptypb.Empty, error) {
|
|
|
|
errs, ctx := errgroup.WithContext(ctx)
|
|
|
|
|
|
|
|
// Broadcasting and saving contribution into the pool in parallel. As one fail should not affect another.
|
|
|
|
errs.Go(func() error {
|
|
|
|
return vs.P2P.Broadcast(ctx, s)
|
|
|
|
})
|
|
|
|
|
|
|
|
if err := vs.SyncCommitteePool.SaveSyncCommitteeContribution(s.Message.Contribution); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Wait for p2p broadcast to complete and return the first error (if any)
|
|
|
|
err := errs.Wait()
|
2021-10-28 08:56:22 +00:00
|
|
|
|
|
|
|
if err == nil {
|
|
|
|
vs.OperationNotifier.OperationFeed().Send(&feed.Event{
|
|
|
|
Type: opfeed.SyncCommitteeContributionReceived,
|
|
|
|
Data: &opfeed.SyncCommitteeContributionReceivedData{
|
|
|
|
Contribution: s,
|
|
|
|
},
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2021-09-01 23:08:05 +00:00
|
|
|
return &emptypb.Empty{}, err
|
|
|
|
}
|
2021-09-03 14:47:35 +00:00
|
|
|
|
|
|
|
// AggregatedSigAndAggregationBits returns the aggregated signature and aggregation bits
|
|
|
|
// associated with a particular set of sync committee messages.
|
|
|
|
func (vs *Server) AggregatedSigAndAggregationBits(
|
|
|
|
ctx context.Context,
|
|
|
|
msgs []*ethpb.SyncCommitteeMessage,
|
|
|
|
slot types.Slot,
|
|
|
|
subnetId uint64,
|
|
|
|
blockRoot []byte,
|
|
|
|
) ([]byte, []byte, error) {
|
|
|
|
subCommitteeSize := params.BeaconConfig().SyncCommitteeSize / params.BeaconConfig().SyncCommitteeSubnetCount
|
2022-01-21 22:57:29 +00:00
|
|
|
sigs := make([][]byte, 0, subCommitteeSize)
|
2021-09-03 14:47:35 +00:00
|
|
|
bits := ethpb.NewSyncCommitteeAggregationBits()
|
|
|
|
for _, msg := range msgs {
|
|
|
|
if bytes.Equal(blockRoot, msg.BlockRoot) {
|
2021-09-20 18:06:43 +00:00
|
|
|
headSyncCommitteeIndices, err := vs.HeadFetcher.HeadSyncCommitteeIndices(ctx, msg.ValidatorIndex, slot)
|
2021-09-03 14:47:35 +00:00
|
|
|
if err != nil {
|
|
|
|
return []byte{}, nil, errors.Wrapf(err, "could not get sync subcommittee index")
|
|
|
|
}
|
2021-09-20 18:06:43 +00:00
|
|
|
for _, index := range headSyncCommitteeIndices {
|
2021-09-03 14:47:35 +00:00
|
|
|
i := uint64(index)
|
|
|
|
subnetIndex := i / subCommitteeSize
|
2022-01-21 18:25:59 +00:00
|
|
|
indexMod := i % subCommitteeSize
|
|
|
|
if subnetIndex == subnetId && !bits.BitAt(indexMod) {
|
|
|
|
bits.SetBitAt(indexMod, true)
|
2022-01-21 22:57:29 +00:00
|
|
|
sigs = append(sigs, msg.Signature)
|
2021-09-03 14:47:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
aggregatedSig := make([]byte, 96)
|
|
|
|
aggregatedSig[0] = 0xC0
|
|
|
|
if len(sigs) != 0 {
|
2022-01-21 22:57:29 +00:00
|
|
|
uncompressedSigs, err := bls.MultipleSignaturesFromBytes(sigs)
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, errors.Wrapf(err, "could not decompress signatures")
|
|
|
|
}
|
|
|
|
aggregatedSig = bls.AggregateSignatures(uncompressedSigs).Marshal()
|
2021-09-03 14:47:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return aggregatedSig, bits, nil
|
|
|
|
}
|