2022-12-15 22:34:05 +00:00
|
|
|
package beacon_api
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2023-01-06 03:32:13 +00:00
|
|
|
"context"
|
2022-12-15 22:34:05 +00:00
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
neturl "net/url"
|
|
|
|
|
|
|
|
"github.com/ethereum/go-ethereum/common/hexutil"
|
|
|
|
"github.com/pkg/errors"
|
2023-03-17 18:52:56 +00:00
|
|
|
"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"
|
2022-12-15 22:34:05 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
type abstractProduceBlockResponseJson struct {
|
|
|
|
Version string `json:"version" enum:"true"`
|
|
|
|
Data json.RawMessage `json:"data"`
|
|
|
|
}
|
|
|
|
|
2023-01-26 14:40:12 +00:00
|
|
|
func (c beaconApiValidatorClient) getBeaconBlock(ctx context.Context, slot primitives.Slot, randaoReveal []byte, graffiti []byte) (*ethpb.GenericBeaconBlock, error) {
|
2022-12-15 22:34:05 +00:00
|
|
|
queryParams := neturl.Values{}
|
|
|
|
queryParams.Add("randao_reveal", hexutil.Encode(randaoReveal))
|
|
|
|
|
|
|
|
if len(graffiti) > 0 {
|
|
|
|
queryParams.Add("graffiti", hexutil.Encode(graffiti))
|
|
|
|
}
|
|
|
|
|
|
|
|
queryUrl := buildURL(fmt.Sprintf("/eth/v2/validator/blocks/%d", slot), queryParams)
|
|
|
|
|
|
|
|
// Since we don't know yet what the json looks like, we unmarshal into an abstract structure that has only a version
|
|
|
|
// and a blob of data
|
|
|
|
produceBlockResponseJson := abstractProduceBlockResponseJson{}
|
2023-01-06 03:32:13 +00:00
|
|
|
if _, err := c.jsonRestHandler.GetRestJsonResponse(ctx, queryUrl, &produceBlockResponseJson); err != nil {
|
2022-12-15 22:34:05 +00:00
|
|
|
return nil, errors.Wrap(err, "failed to query GET REST endpoint")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Once we know what the consensus version is, we can go ahead and unmarshal into the specific structs unique to each version
|
|
|
|
decoder := json.NewDecoder(bytes.NewReader(produceBlockResponseJson.Data))
|
|
|
|
decoder.DisallowUnknownFields()
|
|
|
|
|
|
|
|
response := ðpb.GenericBeaconBlock{}
|
|
|
|
|
|
|
|
switch produceBlockResponseJson.Version {
|
|
|
|
case "phase0":
|
|
|
|
jsonPhase0Block := apimiddleware.BeaconBlockJson{}
|
|
|
|
if err := decoder.Decode(&jsonPhase0Block); err != nil {
|
|
|
|
return nil, errors.Wrap(err, "failed to decode phase0 block response json")
|
|
|
|
}
|
|
|
|
|
2023-02-13 13:22:11 +00:00
|
|
|
phase0Block, err := c.beaconBlockConverter.ConvertRESTPhase0BlockToProto(&jsonPhase0Block)
|
2022-12-15 22:34:05 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, "failed to get phase0 block")
|
|
|
|
}
|
2023-02-13 13:22:11 +00:00
|
|
|
response.Block = ðpb.GenericBeaconBlock_Phase0{
|
|
|
|
Phase0: phase0Block,
|
|
|
|
}
|
2022-12-15 22:34:05 +00:00
|
|
|
|
|
|
|
case "altair":
|
|
|
|
jsonAltairBlock := apimiddleware.BeaconBlockAltairJson{}
|
|
|
|
if err := decoder.Decode(&jsonAltairBlock); err != nil {
|
|
|
|
return nil, errors.Wrap(err, "failed to decode altair block response json")
|
|
|
|
}
|
|
|
|
|
2023-02-13 13:22:11 +00:00
|
|
|
altairBlock, err := c.beaconBlockConverter.ConvertRESTAltairBlockToProto(&jsonAltairBlock)
|
2022-12-15 22:34:05 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, "failed to get altair block")
|
|
|
|
}
|
2023-02-13 13:22:11 +00:00
|
|
|
response.Block = ðpb.GenericBeaconBlock_Altair{
|
|
|
|
Altair: altairBlock,
|
|
|
|
}
|
2022-12-15 22:34:05 +00:00
|
|
|
|
|
|
|
case "bellatrix":
|
|
|
|
jsonBellatrixBlock := apimiddleware.BeaconBlockBellatrixJson{}
|
|
|
|
if err := decoder.Decode(&jsonBellatrixBlock); err != nil {
|
|
|
|
return nil, errors.Wrap(err, "failed to decode bellatrix block response json")
|
|
|
|
}
|
|
|
|
|
2023-02-13 13:22:11 +00:00
|
|
|
bellatrixBlock, err := c.beaconBlockConverter.ConvertRESTBellatrixBlockToProto(&jsonBellatrixBlock)
|
2022-12-15 22:34:05 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, "failed to get bellatrix block")
|
|
|
|
}
|
2023-02-13 13:22:11 +00:00
|
|
|
response.Block = ðpb.GenericBeaconBlock_Bellatrix{
|
|
|
|
Bellatrix: bellatrixBlock,
|
|
|
|
}
|
2022-12-15 22:34:05 +00:00
|
|
|
|
2023-01-11 16:55:09 +00:00
|
|
|
case "capella":
|
|
|
|
jsonCapellaBlock := apimiddleware.BeaconBlockCapellaJson{}
|
|
|
|
if err := decoder.Decode(&jsonCapellaBlock); err != nil {
|
|
|
|
return nil, errors.Wrap(err, "failed to decode capella block response json")
|
|
|
|
}
|
|
|
|
|
2023-02-13 13:22:11 +00:00
|
|
|
capellaBlock, err := c.beaconBlockConverter.ConvertRESTCapellaBlockToProto(&jsonCapellaBlock)
|
2023-01-11 16:55:09 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, "failed to get capella block")
|
|
|
|
}
|
2023-02-13 13:22:11 +00:00
|
|
|
response.Block = ðpb.GenericBeaconBlock_Capella{
|
|
|
|
Capella: capellaBlock,
|
|
|
|
}
|
2023-01-11 16:55:09 +00:00
|
|
|
|
2022-12-15 22:34:05 +00:00
|
|
|
default:
|
|
|
|
return nil, errors.Errorf("unsupported consensus version `%s`", produceBlockResponseJson.Version)
|
|
|
|
}
|
|
|
|
return response, nil
|
|
|
|
}
|