2019-01-28 15:40:40 +00:00
|
|
|
package rpc
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
|
2019-04-17 23:10:55 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/core/blocks"
|
2019-02-13 21:04:31 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/core/helpers"
|
2019-01-28 15:40:40 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/core/state"
|
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/db"
|
|
|
|
pbp2p "github.com/prysmaticlabs/prysm/proto/beacon/p2p/v1"
|
|
|
|
pb "github.com/prysmaticlabs/prysm/proto/beacon/rpc/v1"
|
2019-01-31 02:53:58 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/bytesutil"
|
2019-03-25 15:21:21 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/featureconfig"
|
2019-03-03 17:31:29 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/hashutil"
|
|
|
|
"github.com/prysmaticlabs/prysm/shared/params"
|
2019-01-28 15:40:40 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// ProposerServer defines a server implementation of the gRPC Proposer service,
|
|
|
|
// providing RPC endpoints for computing state transitions and state roots, proposing
|
|
|
|
// beacon blocks to a beacon node, and more.
|
|
|
|
type ProposerServer struct {
|
|
|
|
beaconDB *db.BeaconDB
|
|
|
|
chainService chainService
|
|
|
|
powChainService powChainService
|
2019-02-16 00:36:40 +00:00
|
|
|
operationService operationService
|
2019-01-28 15:40:40 +00:00
|
|
|
canonicalStateChan chan *pbp2p.BeaconState
|
|
|
|
}
|
|
|
|
|
2019-01-28 19:41:04 +00:00
|
|
|
// ProposerIndex sends a response to the client which returns the proposer index for a given slot. Validators
|
|
|
|
// are shuffled and assigned slots to attest/propose to. This method will look for the validator that is assigned
|
|
|
|
// to propose a beacon block at the given slot.
|
|
|
|
func (ps *ProposerServer) ProposerIndex(ctx context.Context, req *pb.ProposerIndexRequest) (*pb.ProposerIndexResponse, error) {
|
2019-04-05 14:41:49 +00:00
|
|
|
beaconState, err := ps.beaconDB.HeadState(ctx)
|
2019-01-28 19:41:04 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("could not get beacon state: %v", err)
|
|
|
|
}
|
|
|
|
|
2019-02-13 21:04:31 +00:00
|
|
|
proposerIndex, err := helpers.BeaconProposerIndex(
|
2019-01-28 19:41:04 +00:00
|
|
|
beaconState,
|
|
|
|
req.SlotNumber,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("could not get index of previous proposer: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return &pb.ProposerIndexResponse{
|
|
|
|
Index: proposerIndex,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
2019-03-13 21:17:32 +00:00
|
|
|
// ProposeBlock is called by a proposer during its assigned slot to create a block in an attempt
|
|
|
|
// to get it processed by the beacon node as the canonical head.
|
2019-01-28 15:40:40 +00:00
|
|
|
func (ps *ProposerServer) ProposeBlock(ctx context.Context, blk *pbp2p.BeaconBlock) (*pb.ProposeResponse, error) {
|
2019-02-26 03:42:31 +00:00
|
|
|
h, err := hashutil.HashBeaconBlock(blk)
|
2019-01-28 15:40:40 +00:00
|
|
|
if err != nil {
|
2019-02-14 20:04:47 +00:00
|
|
|
return nil, fmt.Errorf("could not tree hash block: %v", err)
|
2019-01-28 15:40:40 +00:00
|
|
|
}
|
2019-02-14 20:04:47 +00:00
|
|
|
log.WithField("blockRoot", fmt.Sprintf("%#x", h)).Debugf("Block proposal received via RPC")
|
2019-03-13 21:17:32 +00:00
|
|
|
beaconState, err := ps.chainService.ReceiveBlock(ctx, blk)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("could not process beacon block: %v", err)
|
|
|
|
}
|
2019-04-02 08:49:45 +00:00
|
|
|
if err := ps.beaconDB.UpdateChainHead(ctx, blk, beaconState); err != nil {
|
2019-03-25 15:21:21 +00:00
|
|
|
return nil, fmt.Errorf("failed to update chain: %v", err)
|
|
|
|
}
|
|
|
|
log.WithField("headRoot", fmt.Sprintf("0x%x", h)).Info("Chain head block and state updated")
|
|
|
|
|
2019-04-09 19:06:23 +00:00
|
|
|
if err := ps.beaconDB.SaveHistoricalState(ctx, beaconState); err != nil {
|
2019-03-25 15:21:21 +00:00
|
|
|
log.Errorf("Could not save new historical state: %v", err)
|
2019-03-13 21:17:32 +00:00
|
|
|
}
|
2019-03-18 15:45:28 +00:00
|
|
|
return &pb.ProposeResponse{BlockRootHash32: h[:]}, nil
|
2019-01-28 15:40:40 +00:00
|
|
|
}
|
|
|
|
|
2019-02-16 00:36:40 +00:00
|
|
|
// PendingAttestations retrieves attestations kept in the beacon node's operations pool which have
|
|
|
|
// not yet been included into the beacon chain. Proposers include these pending attestations in their
|
2019-02-25 02:09:45 +00:00
|
|
|
// proposed blocks when performing their responsibility. If desired, callers can choose to filter pending
|
|
|
|
// attestations which are ready for inclusion. That is, attestations that satisfy:
|
|
|
|
// attestation.slot + MIN_ATTESTATION_INCLUSION_DELAY <= state.slot.
|
|
|
|
func (ps *ProposerServer) PendingAttestations(ctx context.Context, req *pb.PendingAttestationsRequest) (*pb.PendingAttestationsResponse, error) {
|
2019-04-05 14:41:49 +00:00
|
|
|
beaconState, err := ps.beaconDB.HeadState(ctx)
|
2019-02-25 02:09:45 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("could not retrieve beacon state: %v", err)
|
|
|
|
}
|
2019-02-16 00:36:40 +00:00
|
|
|
atts, err := ps.operationService.PendingAttestations()
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("could not retrieve pending attestations from operations service: %v", err)
|
|
|
|
}
|
2019-03-06 16:54:02 +00:00
|
|
|
|
2019-03-19 16:25:34 +00:00
|
|
|
head, err := ps.beaconDB.ChainHead()
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to retrieve chain head: %v", err)
|
|
|
|
}
|
|
|
|
blockRoot, err := hashutil.HashBeaconBlock(head)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("could not hash beacon block: %v", err)
|
|
|
|
}
|
|
|
|
for beaconState.Slot < req.ProposalBlockSlot {
|
|
|
|
beaconState, err = state.ExecuteStateTransition(
|
2019-04-10 06:52:06 +00:00
|
|
|
ctx, beaconState, nil /* block */, blockRoot, &state.TransitionConfig{},
|
2019-03-19 16:25:34 +00:00
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("could not execute head transition: %v", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-07 15:56:22 +00:00
|
|
|
// Use the optional proposal block slot parameter as the current slot for
|
|
|
|
// determining the validity window for attestations.
|
|
|
|
currentSlot := req.ProposalBlockSlot
|
|
|
|
if currentSlot == 0 {
|
|
|
|
currentSlot = beaconState.Slot
|
|
|
|
}
|
|
|
|
|
2019-04-17 23:10:55 +00:00
|
|
|
validAtts := make([]*pbp2p.Attestation, 0, len(atts))
|
2019-03-06 16:54:02 +00:00
|
|
|
for _, att := range atts {
|
2019-04-17 23:10:55 +00:00
|
|
|
if err := blocks.VerifyAttestation(beaconState, att, false); err != nil {
|
|
|
|
log.WithError(err).WithField(
|
|
|
|
"slot", att.Data.Slot-params.BeaconConfig().GenesisSlot).Warn(
|
|
|
|
"Skipping, pending attestation failed verification")
|
|
|
|
continue
|
2019-03-19 16:25:34 +00:00
|
|
|
}
|
2019-04-17 23:10:55 +00:00
|
|
|
validAtts = append(validAtts, att)
|
2019-03-19 16:25:34 +00:00
|
|
|
|
2019-03-06 16:54:02 +00:00
|
|
|
}
|
2019-04-17 23:10:55 +00:00
|
|
|
atts = validAtts
|
2019-03-06 16:54:02 +00:00
|
|
|
|
2019-02-25 02:09:45 +00:00
|
|
|
if req.FilterReadyForInclusion {
|
|
|
|
var attsReadyForInclusion []*pbp2p.Attestation
|
|
|
|
for _, val := range atts {
|
2019-03-07 15:56:22 +00:00
|
|
|
if val.Data.Slot+params.BeaconConfig().MinAttestationInclusionDelay <= currentSlot {
|
2019-02-25 02:09:45 +00:00
|
|
|
attsReadyForInclusion = append(attsReadyForInclusion, val)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return &pb.PendingAttestationsResponse{
|
|
|
|
PendingAttestations: attsReadyForInclusion,
|
|
|
|
}, nil
|
|
|
|
}
|
2019-02-16 00:36:40 +00:00
|
|
|
return &pb.PendingAttestationsResponse{
|
|
|
|
PendingAttestations: atts,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
2019-01-28 15:40:40 +00:00
|
|
|
// ComputeStateRoot computes the state root after a block has been processed through a state transition and
|
|
|
|
// returns it to the validator client.
|
|
|
|
func (ps *ProposerServer) ComputeStateRoot(ctx context.Context, req *pbp2p.BeaconBlock) (*pb.StateRootResponse, error) {
|
2019-03-25 15:21:21 +00:00
|
|
|
if !featureconfig.FeatureConfig().EnableComputeStateRoot {
|
|
|
|
log.Debug("Compute state root disabled, returning no-op result")
|
|
|
|
return &pb.StateRootResponse{StateRoot: []byte("no-op")}, nil
|
|
|
|
}
|
|
|
|
|
2019-04-05 14:41:49 +00:00
|
|
|
beaconState, err := ps.beaconDB.HeadState(ctx)
|
2019-01-28 15:40:40 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("could not get beacon state: %v", err)
|
|
|
|
}
|
|
|
|
|
2019-01-31 02:53:58 +00:00
|
|
|
parentHash := bytesutil.ToBytes32(req.ParentRootHash32)
|
2019-02-25 02:09:45 +00:00
|
|
|
// Check for skipped slots.
|
|
|
|
for beaconState.Slot < req.Slot-1 {
|
|
|
|
beaconState, err = state.ExecuteStateTransition(
|
2019-03-05 20:22:09 +00:00
|
|
|
ctx,
|
2019-02-25 02:09:45 +00:00
|
|
|
beaconState,
|
|
|
|
nil,
|
|
|
|
parentHash,
|
2019-03-18 06:19:44 +00:00
|
|
|
state.DefaultConfig(),
|
2019-02-25 02:09:45 +00:00
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("could not execute state transition %v", err)
|
|
|
|
}
|
|
|
|
}
|
2019-01-30 12:45:01 +00:00
|
|
|
beaconState, err = state.ExecuteStateTransition(
|
2019-03-05 20:22:09 +00:00
|
|
|
ctx,
|
2019-01-30 12:45:01 +00:00
|
|
|
beaconState,
|
|
|
|
req,
|
|
|
|
parentHash,
|
2019-03-18 06:19:44 +00:00
|
|
|
state.DefaultConfig(),
|
2019-01-30 12:45:01 +00:00
|
|
|
)
|
2019-01-28 15:40:40 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("could not execute state transition %v", err)
|
|
|
|
}
|
|
|
|
|
2019-02-26 03:42:31 +00:00
|
|
|
beaconStateHash, err := hashutil.HashProto(beaconState)
|
2019-01-28 15:40:40 +00:00
|
|
|
if err != nil {
|
2019-02-14 20:04:47 +00:00
|
|
|
return nil, fmt.Errorf("could not tree hash beacon state: %v", err)
|
2019-01-28 15:40:40 +00:00
|
|
|
}
|
|
|
|
log.WithField("beaconStateHash", fmt.Sprintf("%#x", beaconStateHash)).Debugf("Computed state hash")
|
|
|
|
return &pb.StateRootResponse{
|
|
|
|
StateRoot: beaconStateHash[:],
|
|
|
|
}, nil
|
|
|
|
}
|