2021-09-03 19:11:40 +00:00
|
|
|
package validator
|
|
|
|
|
|
|
|
import (
|
2021-09-18 17:26:11 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/async/event"
|
2021-09-03 19:11:40 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/core/blocks"
|
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/core/feed"
|
|
|
|
blockfeed "github.com/prysmaticlabs/prysm/beacon-chain/core/feed/block"
|
|
|
|
statefeed "github.com/prysmaticlabs/prysm/beacon-chain/core/feed/state"
|
|
|
|
ethpb "github.com/prysmaticlabs/prysm/proto/prysm/v1alpha1"
|
2021-09-16 09:46:29 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/runtime/version"
|
2021-09-03 19:11:40 +00:00
|
|
|
"google.golang.org/grpc/codes"
|
|
|
|
"google.golang.org/grpc/status"
|
|
|
|
)
|
|
|
|
|
|
|
|
// StreamBlocksAltair to clients every single time a block is received by the beacon node.
|
2021-09-24 17:42:16 +00:00
|
|
|
func (vs *Server) StreamBlocksAltair(req *ethpb.StreamBlocksRequest, stream ethpb.BeaconNodeValidator_StreamBlocksAltairServer) error {
|
2021-09-03 19:11:40 +00:00
|
|
|
blocksChannel := make(chan *feed.Event, 1)
|
|
|
|
var blockSub event.Subscription
|
|
|
|
if req.VerifiedOnly {
|
2021-09-24 17:42:16 +00:00
|
|
|
blockSub = vs.StateNotifier.StateFeed().Subscribe(blocksChannel)
|
2021-09-03 19:11:40 +00:00
|
|
|
} else {
|
2021-09-24 17:42:16 +00:00
|
|
|
blockSub = vs.BlockNotifier.BlockFeed().Subscribe(blocksChannel)
|
2021-09-03 19:11:40 +00:00
|
|
|
}
|
|
|
|
defer blockSub.Unsubscribe()
|
|
|
|
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case blockEvent := <-blocksChannel:
|
|
|
|
if req.VerifiedOnly {
|
|
|
|
if blockEvent.Type == statefeed.BlockProcessed {
|
|
|
|
data, ok := blockEvent.Data.(*statefeed.BlockProcessedData)
|
|
|
|
if !ok || data == nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
b := ðpb.StreamBlocksResponse{}
|
|
|
|
switch data.SignedBlock.Version() {
|
|
|
|
case version.Phase0:
|
|
|
|
phBlk, ok := data.SignedBlock.Proto().(*ethpb.SignedBeaconBlock)
|
|
|
|
if !ok {
|
|
|
|
log.Warn("Mismatch between version and block type, was expecting *ethpb.SignedBeaconBlock")
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
b.Block = ðpb.StreamBlocksResponse_Phase0Block{Phase0Block: phBlk}
|
|
|
|
case version.Altair:
|
|
|
|
phBlk, ok := data.SignedBlock.Proto().(*ethpb.SignedBeaconBlockAltair)
|
|
|
|
if !ok {
|
|
|
|
log.Warn("Mismatch between version and block type, was expecting *v2.SignedBeaconBlockAltair")
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
b.Block = ðpb.StreamBlocksResponse_AltairBlock{AltairBlock: phBlk}
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := stream.Send(b); err != nil {
|
|
|
|
return status.Errorf(codes.Unavailable, "Could not send over stream: %v", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if blockEvent.Type == blockfeed.ReceivedBlock {
|
|
|
|
data, ok := blockEvent.Data.(*blockfeed.ReceivedBlockData)
|
|
|
|
if !ok {
|
|
|
|
// Got bad data over the stream.
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if data.SignedBlock == nil {
|
|
|
|
// One nil block shouldn't stop the stream.
|
|
|
|
continue
|
|
|
|
}
|
2021-09-24 17:42:16 +00:00
|
|
|
headState, err := vs.HeadFetcher.HeadState(vs.Ctx)
|
2021-09-03 19:11:40 +00:00
|
|
|
if err != nil {
|
|
|
|
log.WithError(err).WithField("blockSlot", data.SignedBlock.Block().Slot()).Error("Could not get head state")
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
signed := data.SignedBlock
|
|
|
|
if err := blocks.VerifyBlockSignature(headState, signed.Block().ProposerIndex(), signed.Signature(), signed.Block().HashTreeRoot); err != nil {
|
|
|
|
log.WithError(err).WithField("blockSlot", data.SignedBlock.Block().Slot()).Error("Could not verify block signature")
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
b := ðpb.StreamBlocksResponse{}
|
|
|
|
switch data.SignedBlock.Version() {
|
|
|
|
case version.Phase0:
|
|
|
|
phBlk, ok := data.SignedBlock.Proto().(*ethpb.SignedBeaconBlock)
|
|
|
|
if !ok {
|
|
|
|
log.Warn("Mismatch between version and block type, was expecting *ethpb.SignedBeaconBlock")
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
b.Block = ðpb.StreamBlocksResponse_Phase0Block{Phase0Block: phBlk}
|
|
|
|
case version.Altair:
|
|
|
|
phBlk, ok := data.SignedBlock.Proto().(*ethpb.SignedBeaconBlockAltair)
|
|
|
|
if !ok {
|
|
|
|
log.Warn("Mismatch between version and block type, was expecting *v2.SignedBeaconBlockAltair")
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
b.Block = ðpb.StreamBlocksResponse_AltairBlock{AltairBlock: phBlk}
|
|
|
|
}
|
|
|
|
if err := stream.Send(b); err != nil {
|
|
|
|
return status.Errorf(codes.Unavailable, "Could not send over stream: %v", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case <-blockSub.Err():
|
|
|
|
return status.Error(codes.Aborted, "Subscriber closed, exiting goroutine")
|
2021-09-24 17:42:16 +00:00
|
|
|
case <-vs.Ctx.Done():
|
2021-09-03 19:11:40 +00:00
|
|
|
return status.Error(codes.Canceled, "Context canceled")
|
|
|
|
case <-stream.Context().Done():
|
|
|
|
return status.Error(codes.Canceled, "Context canceled")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|