mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-14 14:08:19 +00:00
d17996f8b0
* Update V3 from V4 * Fix build v3 -> v4 * Update ssz * Update beacon_chain.pb.go * Fix formatter import * Update update-mockgen.sh comment to v4 * Fix conflicts. Pass build and tests * Fix test
166 lines
5.9 KiB
Go
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/v4/beacon-chain/rpc/apimiddleware"
|
|
"github.com/prysmaticlabs/prysm/v4/consensus-types/primitives"
|
|
ethpb "github.com/prysmaticlabs/prysm/v4/proto/prysm/v1alpha1"
|
|
"github.com/prysmaticlabs/prysm/v4/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 ðpb.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, ðpb.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 ðpb.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 ðpb.SyncCommitteeContribution{
|
|
Slot: primitives.Slot(slot),
|
|
BlockRoot: blockRoot,
|
|
SubcommitteeIndex: subcommitteeIdx,
|
|
AggregationBits: aggregationBits,
|
|
Signature: signature,
|
|
}, nil
|
|
}
|