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
197 lines
6.6 KiB
Go
197 lines
6.6 KiB
Go
package beacon_api
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"encoding/json"
|
|
"time"
|
|
|
|
"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"
|
|
"google.golang.org/grpc"
|
|
)
|
|
|
|
type abstractSignedBlockResponseJson struct {
|
|
Version string `json:"version" enum:"true"`
|
|
ExecutionOptimistic bool `json:"execution_optimistic"`
|
|
Finalized bool `json:"finalized"`
|
|
Data json.RawMessage `json:"data"`
|
|
}
|
|
|
|
type streamBlocksAltairClient struct {
|
|
grpc.ClientStream
|
|
ctx context.Context
|
|
beaconApiClient beaconApiValidatorClient
|
|
streamBlocksRequest *ethpb.StreamBlocksRequest
|
|
prevBlockSlot primitives.Slot
|
|
pingDelay time.Duration
|
|
}
|
|
|
|
type headSignedBeaconBlockResult struct {
|
|
streamBlocksResponse *ethpb.StreamBlocksResponse
|
|
executionOptimistic bool
|
|
slot primitives.Slot
|
|
}
|
|
|
|
func (c beaconApiValidatorClient) streamBlocks(ctx context.Context, in *ethpb.StreamBlocksRequest, pingDelay time.Duration) ethpb.BeaconNodeValidator_StreamBlocksAltairClient {
|
|
return &streamBlocksAltairClient{
|
|
ctx: ctx,
|
|
beaconApiClient: c,
|
|
streamBlocksRequest: in,
|
|
pingDelay: pingDelay,
|
|
}
|
|
}
|
|
|
|
func (c *streamBlocksAltairClient) Recv() (*ethpb.StreamBlocksResponse, error) {
|
|
result, err := c.beaconApiClient.getHeadSignedBeaconBlock(c.ctx)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "failed to get latest signed block")
|
|
}
|
|
|
|
// We keep querying the beacon chain for the latest block until we receive a new slot
|
|
for (c.streamBlocksRequest.VerifiedOnly && result.executionOptimistic) || c.prevBlockSlot == result.slot {
|
|
select {
|
|
case <-time.After(c.pingDelay):
|
|
result, err = c.beaconApiClient.getHeadSignedBeaconBlock(c.ctx)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "failed to get latest signed block")
|
|
}
|
|
case <-c.ctx.Done():
|
|
return nil, errors.New("context canceled")
|
|
}
|
|
}
|
|
|
|
c.prevBlockSlot = result.slot
|
|
return result.streamBlocksResponse, nil
|
|
}
|
|
|
|
func (c beaconApiValidatorClient) getHeadSignedBeaconBlock(ctx context.Context) (*headSignedBeaconBlockResult, error) {
|
|
// 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
|
|
signedBlockResponseJson := abstractSignedBlockResponseJson{}
|
|
if _, err := c.jsonRestHandler.GetRestJsonResponse(ctx, "/eth/v2/beacon/blocks/head", &signedBlockResponseJson); err != nil {
|
|
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(signedBlockResponseJson.Data))
|
|
decoder.DisallowUnknownFields()
|
|
|
|
response := ðpb.StreamBlocksResponse{}
|
|
var slot primitives.Slot
|
|
|
|
switch signedBlockResponseJson.Version {
|
|
case "phase0":
|
|
jsonPhase0Block := apimiddleware.SignedBeaconBlockContainerJson{}
|
|
if err := decoder.Decode(&jsonPhase0Block); err != nil {
|
|
return nil, errors.Wrap(err, "failed to decode signed phase0 block response json")
|
|
}
|
|
|
|
phase0Block, err := c.beaconBlockConverter.ConvertRESTPhase0BlockToProto(jsonPhase0Block.Message)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "failed to get signed phase0 block")
|
|
}
|
|
|
|
decodedSignature, err := hexutil.Decode(jsonPhase0Block.Signature)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "failed to decode phase0 block signature `%s`", jsonPhase0Block.Signature)
|
|
}
|
|
|
|
response.Block = ðpb.StreamBlocksResponse_Phase0Block{
|
|
Phase0Block: ðpb.SignedBeaconBlock{
|
|
Signature: decodedSignature,
|
|
Block: phase0Block,
|
|
},
|
|
}
|
|
|
|
slot = phase0Block.Slot
|
|
|
|
case "altair":
|
|
jsonAltairBlock := apimiddleware.SignedBeaconBlockAltairContainerJson{}
|
|
if err := decoder.Decode(&jsonAltairBlock); err != nil {
|
|
return nil, errors.Wrap(err, "failed to decode signed altair block response json")
|
|
}
|
|
|
|
altairBlock, err := c.beaconBlockConverter.ConvertRESTAltairBlockToProto(jsonAltairBlock.Message)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "failed to get signed altair block")
|
|
}
|
|
|
|
decodedSignature, err := hexutil.Decode(jsonAltairBlock.Signature)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "failed to decode altair block signature `%s`", jsonAltairBlock.Signature)
|
|
}
|
|
|
|
response.Block = ðpb.StreamBlocksResponse_AltairBlock{
|
|
AltairBlock: ðpb.SignedBeaconBlockAltair{
|
|
Signature: decodedSignature,
|
|
Block: altairBlock,
|
|
},
|
|
}
|
|
|
|
slot = altairBlock.Slot
|
|
|
|
case "bellatrix":
|
|
jsonBellatrixBlock := apimiddleware.SignedBeaconBlockBellatrixContainerJson{}
|
|
if err := decoder.Decode(&jsonBellatrixBlock); err != nil {
|
|
return nil, errors.Wrap(err, "failed to decode signed bellatrix block response json")
|
|
}
|
|
|
|
bellatrixBlock, err := c.beaconBlockConverter.ConvertRESTBellatrixBlockToProto(jsonBellatrixBlock.Message)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "failed to get signed bellatrix block")
|
|
}
|
|
|
|
decodedSignature, err := hexutil.Decode(jsonBellatrixBlock.Signature)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "failed to decode bellatrix block signature `%s`", jsonBellatrixBlock.Signature)
|
|
}
|
|
|
|
response.Block = ðpb.StreamBlocksResponse_BellatrixBlock{
|
|
BellatrixBlock: ðpb.SignedBeaconBlockBellatrix{
|
|
Signature: decodedSignature,
|
|
Block: bellatrixBlock,
|
|
},
|
|
}
|
|
|
|
slot = bellatrixBlock.Slot
|
|
|
|
case "capella":
|
|
jsonCapellaBlock := apimiddleware.SignedBeaconBlockCapellaContainerJson{}
|
|
if err := decoder.Decode(&jsonCapellaBlock); err != nil {
|
|
return nil, errors.Wrap(err, "failed to decode signed capella block response json")
|
|
}
|
|
|
|
capellaBlock, err := c.beaconBlockConverter.ConvertRESTCapellaBlockToProto(jsonCapellaBlock.Message)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "failed to get signed capella block")
|
|
}
|
|
|
|
decodedSignature, err := hexutil.Decode(jsonCapellaBlock.Signature)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "failed to decode capella block signature `%s`", jsonCapellaBlock.Signature)
|
|
}
|
|
|
|
response.Block = ðpb.StreamBlocksResponse_CapellaBlock{
|
|
CapellaBlock: ðpb.SignedBeaconBlockCapella{
|
|
Signature: decodedSignature,
|
|
Block: capellaBlock,
|
|
},
|
|
}
|
|
|
|
slot = capellaBlock.Slot
|
|
|
|
default:
|
|
return nil, errors.Errorf("unsupported consensus version `%s`", signedBlockResponseJson.Version)
|
|
}
|
|
|
|
return &headSignedBeaconBlockResult{
|
|
streamBlocksResponse: response,
|
|
executionOptimistic: signedBlockResponseJson.ExecutionOptimistic,
|
|
slot: slot,
|
|
}, nil
|
|
}
|