2021-08-14 16:41:03 +00:00
|
|
|
package beacon
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2021-08-20 21:32:56 +00:00
|
|
|
"fmt"
|
2021-08-14 16:41:03 +00:00
|
|
|
|
|
|
|
"github.com/golang/protobuf/ptypes/empty"
|
2021-09-02 16:54:53 +00:00
|
|
|
"github.com/pkg/errors"
|
2021-08-20 21:32:56 +00:00
|
|
|
types "github.com/prysmaticlabs/eth2-types"
|
2021-09-16 19:55:51 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/api/grpc"
|
2021-08-20 21:32:56 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/core/altair"
|
2021-09-02 16:54:53 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/rpc/eth/helpers"
|
2021-08-20 21:32:56 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/state"
|
2021-09-21 19:59:25 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/config/params"
|
2021-09-23 15:23:37 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/encoding/bytesutil"
|
2021-08-14 16:41:03 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/proto/eth/v2"
|
2021-09-02 16:54:53 +00:00
|
|
|
ethpbv2 "github.com/prysmaticlabs/prysm/proto/eth/v2"
|
|
|
|
ethpbalpha "github.com/prysmaticlabs/prysm/proto/prysm/v1alpha1"
|
|
|
|
"go.opencensus.io/trace"
|
2021-08-20 21:32:56 +00:00
|
|
|
"google.golang.org/grpc/codes"
|
|
|
|
"google.golang.org/grpc/status"
|
2021-08-14 16:41:03 +00:00
|
|
|
)
|
|
|
|
|
2021-08-20 21:32:56 +00:00
|
|
|
// ListSyncCommittees retrieves the sync committees for the given epoch.
|
|
|
|
// If the epoch is not passed in, then the sync committees for the epoch of the state will be obtained.
|
2021-09-02 16:54:53 +00:00
|
|
|
func (bs *Server) ListSyncCommittees(ctx context.Context, req *ethpbv2.StateSyncCommitteesRequest) (*ethpbv2.StateSyncCommitteesResponse, error) {
|
|
|
|
ctx, span := trace.StartSpan(ctx, "beacon.ListSyncCommittees")
|
|
|
|
defer span.End()
|
|
|
|
|
2021-08-20 21:32:56 +00:00
|
|
|
st, err := bs.stateFromRequest(ctx, &stateRequest{
|
|
|
|
epoch: req.Epoch,
|
|
|
|
stateId: req.StateId,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return nil, status.Errorf(codes.Internal, "Could not fetch beacon state using request: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the current sync committee and sync committee indices from the state.
|
|
|
|
committeeIndices, committee, err := currentCommitteeIndicesFromState(st)
|
|
|
|
if err != nil {
|
|
|
|
return nil, status.Errorf(codes.Internal, "Could not get sync committee indices from state: %v", err)
|
|
|
|
}
|
|
|
|
subcommittees, err := extractSyncSubcommittees(st, committee)
|
|
|
|
if err != nil {
|
|
|
|
return nil, status.Errorf(codes.Internal, "Could not extract sync subcommittees: %v", err)
|
|
|
|
}
|
|
|
|
|
2021-09-02 16:54:53 +00:00
|
|
|
return ðpbv2.StateSyncCommitteesResponse{
|
|
|
|
Data: ðpbv2.SyncCommitteeValidators{
|
2021-08-20 21:32:56 +00:00
|
|
|
Validators: committeeIndices,
|
|
|
|
ValidatorAggregates: subcommittees,
|
|
|
|
},
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
2021-09-02 16:54:53 +00:00
|
|
|
func currentCommitteeIndicesFromState(st state.BeaconState) ([]types.ValidatorIndex, *ethpbalpha.SyncCommittee, error) {
|
2021-08-20 21:32:56 +00:00
|
|
|
committee, err := st.CurrentSyncCommittee()
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, fmt.Errorf(
|
|
|
|
"could not get sync committee: %v", err,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
committeeIndices := make([]types.ValidatorIndex, len(committee.Pubkeys))
|
|
|
|
for i, key := range committee.Pubkeys {
|
|
|
|
index, ok := st.ValidatorIndexByPubkey(bytesutil.ToBytes48(key))
|
|
|
|
if !ok {
|
|
|
|
return nil, nil, fmt.Errorf(
|
|
|
|
"validator index not found for pubkey %#x",
|
|
|
|
bytesutil.Trunc(key),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
committeeIndices[i] = index
|
|
|
|
}
|
|
|
|
return committeeIndices, committee, nil
|
2021-08-14 16:41:03 +00:00
|
|
|
}
|
|
|
|
|
2021-09-02 16:54:53 +00:00
|
|
|
func extractSyncSubcommittees(st state.BeaconState, committee *ethpbalpha.SyncCommittee) ([]*eth.SyncSubcommitteeValidators, error) {
|
2021-08-20 21:32:56 +00:00
|
|
|
subcommitteeCount := params.BeaconConfig().SyncCommitteeSubnetCount
|
2021-09-02 16:54:53 +00:00
|
|
|
subcommittees := make([]*ethpbv2.SyncSubcommitteeValidators, subcommitteeCount)
|
2021-08-20 21:32:56 +00:00
|
|
|
for i := uint64(0); i < subcommitteeCount; i++ {
|
|
|
|
pubkeys, err := altair.SyncSubCommitteePubkeys(committee, types.CommitteeIndex(i))
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf(
|
|
|
|
"failed to get subcommittee pubkeys: %v", err,
|
|
|
|
)
|
|
|
|
}
|
2021-09-02 16:54:53 +00:00
|
|
|
subcommittee := ðpbv2.SyncSubcommitteeValidators{Validators: make([]types.ValidatorIndex, len(pubkeys))}
|
2021-08-20 21:32:56 +00:00
|
|
|
for j, key := range pubkeys {
|
|
|
|
index, ok := st.ValidatorIndexByPubkey(bytesutil.ToBytes48(key))
|
|
|
|
if !ok {
|
|
|
|
return nil, fmt.Errorf(
|
|
|
|
"validator index not found for pubkey %#x",
|
|
|
|
bytesutil.Trunc(key),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
subcommittee.Validators[j] = index
|
|
|
|
}
|
|
|
|
subcommittees[i] = subcommittee
|
|
|
|
}
|
|
|
|
return subcommittees, nil
|
2021-08-14 16:41:03 +00:00
|
|
|
}
|
2021-09-02 16:54:53 +00:00
|
|
|
|
|
|
|
// SubmitPoolSyncCommitteeSignatures submits sync committee signature objects to the node.
|
|
|
|
func (bs *Server) SubmitPoolSyncCommitteeSignatures(ctx context.Context, req *ethpbv2.SubmitPoolSyncCommitteeSignatures) (*empty.Empty, error) {
|
|
|
|
ctx, span := trace.StartSpan(ctx, "beacon.SubmitPoolSyncCommitteeSignatures")
|
|
|
|
defer span.End()
|
|
|
|
|
|
|
|
var validMessages []*ethpbalpha.SyncCommitteeMessage
|
|
|
|
var msgFailures []*helpers.SingleIndexedVerificationFailure
|
|
|
|
for i, msg := range req.Data {
|
|
|
|
if err := validateSyncCommitteeMessage(msg); err != nil {
|
|
|
|
msgFailures = append(msgFailures, &helpers.SingleIndexedVerificationFailure{
|
|
|
|
Index: i,
|
|
|
|
Message: err.Error(),
|
|
|
|
})
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
v1alpha1Msg := ðpbalpha.SyncCommitteeMessage{
|
|
|
|
Slot: msg.Slot,
|
|
|
|
BlockRoot: msg.BeaconBlockRoot,
|
|
|
|
ValidatorIndex: msg.ValidatorIndex,
|
|
|
|
Signature: msg.Signature,
|
|
|
|
}
|
|
|
|
validMessages = append(validMessages, v1alpha1Msg)
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, msg := range validMessages {
|
|
|
|
_, err := bs.V1Alpha1ValidatorServer.SubmitSyncMessage(ctx, msg)
|
|
|
|
if err != nil {
|
|
|
|
return nil, status.Errorf(codes.Internal, "Could not submit message: %v", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(msgFailures) > 0 {
|
|
|
|
failuresContainer := &helpers.IndexedVerificationFailure{Failures: msgFailures}
|
2021-09-16 19:55:51 +00:00
|
|
|
err := grpc.AppendCustomErrorHeader(ctx, failuresContainer)
|
2021-09-02 16:54:53 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, status.Errorf(
|
|
|
|
codes.InvalidArgument,
|
|
|
|
"One or more messages failed validation. Could not prepare detailed failure information: %v",
|
|
|
|
err,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
return nil, status.Errorf(codes.InvalidArgument, "One or more messages failed validation")
|
|
|
|
}
|
|
|
|
|
|
|
|
return &empty.Empty{}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func validateSyncCommitteeMessage(msg *ethpbv2.SyncCommitteeMessage) error {
|
|
|
|
if !bytesutil.IsHexOfLen(msg.BeaconBlockRoot, 64) {
|
|
|
|
return errors.New("invalid block root format")
|
|
|
|
}
|
|
|
|
if !bytesutil.IsHexOfLen(msg.Signature, 192) {
|
|
|
|
return errors.New("invalid signature format")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|