prysm-pulse/validator/client/beacon-api/sync_committee.go
Dhruv Bodani 5092738be6
add REST implementation for GetSyncSubcommitteeIndex (#11971)
* add implementation for GetSyncSubcommitteeIndex

* fix goimports

* update error parsing for validator sync committee index

Co-authored-by: Radosław Kapka <radek@prysmaticlabs.com>

* fix imports

---------

Co-authored-by: Radosław Kapka <radek@prysmaticlabs.com>
Co-authored-by: Radosław Kapka <rkapka@wp.pl>
2023-02-09 10:03:32 +00:00

166 lines
5.9 KiB
Go

package beacon_api
import (
"bytes"
"context"
"encoding/json"
"net/url"
"strconv"
"github.com/ethereum/go-ethereum/common/hexutil"
"github.com/pkg/errors"
"github.com/prysmaticlabs/prysm/v3/beacon-chain/rpc/apimiddleware"
"github.com/prysmaticlabs/prysm/v3/consensus-types/primitives"
ethpb "github.com/prysmaticlabs/prysm/v3/proto/prysm/v1alpha1"
"github.com/prysmaticlabs/prysm/v3/time/slots"
)
func (c *beaconApiValidatorClient) submitSyncMessage(ctx context.Context, syncMessage *ethpb.SyncCommitteeMessage) error {
const endpoint = "/eth/v1/beacon/pool/sync_committees"
jsonSyncCommitteeMessage := &apimiddleware.SyncCommitteeMessageJson{
Slot: strconv.FormatUint(uint64(syncMessage.Slot), 10),
BeaconBlockRoot: hexutil.Encode(syncMessage.BlockRoot),
ValidatorIndex: strconv.FormatUint(uint64(syncMessage.ValidatorIndex), 10),
Signature: hexutil.Encode(syncMessage.Signature),
}
marshalledJsonSyncCommitteeMessage, err := json.Marshal([]*apimiddleware.SyncCommitteeMessageJson{jsonSyncCommitteeMessage})
if err != nil {
return errors.Wrap(err, "failed to marshal sync committee message")
}
if _, err := c.jsonRestHandler.PostRestJson(ctx, endpoint, nil, bytes.NewBuffer(marshalledJsonSyncCommitteeMessage), nil); err != nil {
return errors.Wrapf(err, "failed to send POST data to `%s` REST endpoint", endpoint)
}
return nil
}
func (c *beaconApiValidatorClient) getSyncMessageBlockRoot(ctx context.Context) (*ethpb.SyncMessageBlockRootResponse, error) {
// Get head beacon block root.
var resp apimiddleware.BlockRootResponseJson
if _, err := c.jsonRestHandler.GetRestJsonResponse(ctx, "/eth/v1/beacon/blocks/head/root", &resp); err != nil {
return nil, errors.Wrap(err, "failed to query GET REST endpoint")
}
// An optimistic validator MUST NOT participate in sync committees
// (i.e., sign across the DOMAIN_SYNC_COMMITTEE, DOMAIN_SYNC_COMMITTEE_SELECTION_PROOF or DOMAIN_CONTRIBUTION_AND_PROOF domains).
if resp.ExecutionOptimistic {
return nil, errors.New("the node is currently optimistic and cannot serve validators")
}
if resp.Data == nil {
return nil, errors.New("no data returned")
}
if resp.Data.Root == "" {
return nil, errors.New("no root returned")
}
blockRoot, err := hexutil.Decode(resp.Data.Root)
if err != nil {
return nil, errors.Wrap(err, "failed to decode beacon block root")
}
return &ethpb.SyncMessageBlockRootResponse{
Root: blockRoot,
}, nil
}
func (c *beaconApiValidatorClient) getSyncCommitteeContribution(
ctx context.Context,
req *ethpb.SyncCommitteeContributionRequest,
) (*ethpb.SyncCommitteeContribution, error) {
blockRootResponse, err := c.getSyncMessageBlockRoot(ctx)
if err != nil {
return nil, errors.Wrap(err, "failed to get sync message block root")
}
blockRoot := hexutil.Encode(blockRootResponse.Root)
params := url.Values{}
params.Add("slot", strconv.FormatUint(uint64(req.Slot), 10))
params.Add("subcommittee_index", strconv.FormatUint(req.SubnetId, 10))
params.Add("beacon_block_root", blockRoot)
url := buildURL("/eth/v1/validator/sync_committee_contribution", params)
var resp apimiddleware.ProduceSyncCommitteeContributionResponseJson
if _, err := c.jsonRestHandler.GetRestJsonResponse(ctx, url, &resp); err != nil {
return nil, errors.Wrap(err, "failed to query GET REST endpoint")
}
return convertSyncContributionJsonToProto(resp.Data)
}
func (c *beaconApiValidatorClient) getSyncSubcommitteeIndex(ctx context.Context, in *ethpb.SyncSubcommitteeIndexRequest) (*ethpb.SyncSubcommitteeIndexResponse, error) {
validatorIndexResponse, err := c.validatorIndex(ctx, &ethpb.ValidatorIndexRequest{PublicKey: in.PublicKey})
if err != nil {
return nil, errors.Wrap(err, "failed to get validator index")
}
syncDuties, err := c.dutiesProvider.GetSyncDuties(ctx, slots.ToEpoch(in.Slot), []primitives.ValidatorIndex{validatorIndexResponse.Index})
if err != nil {
return nil, errors.Wrap(err, "failed to get sync committee duties")
}
if len(syncDuties) == 0 {
return nil, errors.Errorf("no sync committee duty for the given slot %d", in.Slot)
}
// First sync duty is required since we requested sync duties for one validator index.
syncDuty := syncDuties[0]
var indices []primitives.CommitteeIndex
for _, idx := range syncDuty.ValidatorSyncCommitteeIndices {
syncCommIdx, err := strconv.ParseUint(idx, 10, 64)
if err != nil {
return nil, errors.Wrapf(err, "failed to parse validator sync committee index %s", idx)
}
indices = append(indices, primitives.CommitteeIndex(syncCommIdx))
}
return &ethpb.SyncSubcommitteeIndexResponse{Indices: indices}, nil
}
func convertSyncContributionJsonToProto(contribution *apimiddleware.SyncCommitteeContributionJson) (*ethpb.SyncCommitteeContribution, error) {
if contribution == nil {
return nil, errors.New("sync committee contribution is nil")
}
slot, err := strconv.ParseUint(contribution.Slot, 10, 64)
if err != nil {
return nil, errors.Wrapf(err, "failed to parse slot `%s`", contribution.Slot)
}
blockRoot, err := hexutil.Decode(contribution.BeaconBlockRoot)
if err != nil {
return nil, errors.Wrapf(err, "failed to decode beacon block root `%s`", contribution.BeaconBlockRoot)
}
subcommitteeIdx, err := strconv.ParseUint(contribution.SubcommitteeIndex, 10, 64)
if err != nil {
return nil, errors.Wrapf(err, "failed to parse subcommittee index `%s`", contribution.SubcommitteeIndex)
}
aggregationBits, err := hexutil.Decode(contribution.AggregationBits)
if err != nil {
return nil, errors.Wrapf(err, "failed to decode aggregation bits `%s`", contribution.AggregationBits)
}
signature, err := hexutil.Decode(contribution.Signature)
if err != nil {
return nil, errors.Wrapf(err, "failed to decode contribution signature `%s`", contribution.Signature)
}
return &ethpb.SyncCommitteeContribution{
Slot: primitives.Slot(slot),
BlockRoot: blockRoot,
SubcommitteeIndex: subcommitteeIdx,
AggregationBits: aggregationBits,
Signature: signature,
}, nil
}