2019-01-28 15:40:40 +00:00
|
|
|
package rpc
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
|
2019-05-09 00:27:29 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/cache"
|
2019-02-11 16:15:25 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/core/blocks"
|
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/core/helpers"
|
2019-03-02 00:33:55 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/core/state"
|
2019-02-11 16:15:25 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/db"
|
2019-02-06 16:20:38 +00:00
|
|
|
pbp2p "github.com/prysmaticlabs/prysm/proto/beacon/p2p/v1"
|
2019-01-28 15:40:40 +00:00
|
|
|
pb "github.com/prysmaticlabs/prysm/proto/beacon/rpc/v1"
|
2019-05-07 22:31:06 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/bytesutil"
|
2019-01-28 15:40:40 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/hashutil"
|
2019-05-06 20:30:29 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/p2p"
|
2019-02-15 23:19:36 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/params"
|
2019-01-28 15:40:40 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// AttesterServer defines a server implementation of the gRPC Attester service,
|
|
|
|
// providing RPC methods for validators acting as attesters to broadcast votes on beacon blocks.
|
|
|
|
type AttesterServer struct {
|
2019-05-06 20:30:29 +00:00
|
|
|
p2p p2p.Broadcaster
|
2019-02-11 16:15:25 +00:00
|
|
|
beaconDB *db.BeaconDB
|
2019-01-31 18:54:24 +00:00
|
|
|
operationService operationService
|
2019-05-09 00:27:29 +00:00
|
|
|
cache *cache.AttestationCache
|
2019-01-28 15:40:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// AttestHead is a function called by an attester in a sharding validator to vote
|
|
|
|
// on a block via an attestation object as defined in the Ethereum Serenity specification.
|
2019-02-06 16:20:38 +00:00
|
|
|
func (as *AttesterServer) AttestHead(ctx context.Context, att *pbp2p.Attestation) (*pb.AttestResponse, error) {
|
|
|
|
h, err := hashutil.HashProto(att)
|
2019-01-28 15:40:40 +00:00
|
|
|
if err != nil {
|
2019-01-28 19:41:04 +00:00
|
|
|
return nil, fmt.Errorf("could not hash attestation: %v", err)
|
2019-01-28 15:40:40 +00:00
|
|
|
}
|
2019-03-17 02:56:05 +00:00
|
|
|
|
|
|
|
if err := as.operationService.HandleAttestations(ctx, att); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2019-05-07 22:31:06 +00:00
|
|
|
|
|
|
|
// Update attestation target for RPC server to run necessary fork choice.
|
|
|
|
// We need to retrieve the head block to get its parent root.
|
|
|
|
blk, err := as.beaconDB.Block(bytesutil.ToBytes32(att.Data.BeaconBlockRootHash32))
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
attTarget := &pbp2p.AttestationTarget{
|
|
|
|
Slot: att.Data.Slot,
|
|
|
|
BlockRoot: att.Data.BeaconBlockRootHash32,
|
|
|
|
ParentRoot: blk.ParentRootHash32,
|
|
|
|
}
|
|
|
|
if err := as.beaconDB.SaveAttestationTarget(ctx, attTarget); err != nil {
|
|
|
|
return nil, fmt.Errorf("could not save attestation target")
|
|
|
|
}
|
|
|
|
|
2019-05-06 20:30:29 +00:00
|
|
|
as.p2p.Broadcast(ctx, &pbp2p.AttestationAnnounce{
|
|
|
|
Hash: h[:],
|
|
|
|
})
|
2019-01-28 15:40:40 +00:00
|
|
|
return &pb.AttestResponse{AttestationHash: h[:]}, nil
|
|
|
|
}
|
2019-02-06 16:20:38 +00:00
|
|
|
|
2019-02-27 20:21:15 +00:00
|
|
|
// AttestationDataAtSlot fetches the necessary information from the current canonical head
|
2019-02-11 16:15:25 +00:00
|
|
|
// and beacon state for an assigned attester to perform necessary responsibilities. This includes
|
|
|
|
// fetching the epoch boundary roots, the latest justified block root, among others.
|
2019-02-27 20:21:15 +00:00
|
|
|
func (as *AttesterServer) AttestationDataAtSlot(ctx context.Context, req *pb.AttestationDataRequest) (*pb.AttestationDataResponse, error) {
|
2019-05-09 00:27:29 +00:00
|
|
|
res, err := as.cache.Get(ctx, req)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if res != nil {
|
|
|
|
return res, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := as.cache.MarkInProgress(req); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
defer func() {
|
|
|
|
if err := as.cache.MarkNotInProgress(req); err != nil {
|
|
|
|
log.WithError(err).Error("Failed to mark cache not in progress")
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
2019-02-11 16:15:25 +00:00
|
|
|
// Set the attestation data's beacon block root = hash_tree_root(head) where head
|
|
|
|
// is the validator's view of the head block of the beacon chain during the slot.
|
2019-02-19 05:49:56 +00:00
|
|
|
head, err := as.beaconDB.ChainHead()
|
2019-02-11 16:15:25 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to retrieve chain head: %v", err)
|
|
|
|
}
|
2019-04-05 14:41:49 +00:00
|
|
|
headRoot, err := hashutil.HashBeaconBlock(head)
|
2019-02-11 16:15:25 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("could not tree hash beacon block: %v", err)
|
|
|
|
}
|
2019-04-05 14:41:49 +00:00
|
|
|
|
|
|
|
// Let head state be the state of head block processed through empty slots up to assigned slot.
|
|
|
|
headState, err := as.beaconDB.HeadState(ctx)
|
2019-02-11 16:15:25 +00:00
|
|
|
if err != nil {
|
2019-04-05 14:41:49 +00:00
|
|
|
return nil, fmt.Errorf("could not fetch head state: %v", err)
|
2019-02-11 16:15:25 +00:00
|
|
|
}
|
2019-04-05 14:41:49 +00:00
|
|
|
|
|
|
|
for headState.Slot < req.Slot {
|
2019-05-08 23:51:00 +00:00
|
|
|
if ctx.Err() != nil {
|
|
|
|
return nil, ctx.Err()
|
|
|
|
}
|
|
|
|
|
2019-04-05 14:41:49 +00:00
|
|
|
headState, err = state.ExecuteStateTransition(
|
2019-04-10 06:52:06 +00:00
|
|
|
ctx, headState, nil /* block */, headRoot, state.DefaultConfig(),
|
2019-03-02 00:33:55 +00:00
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("could not execute head transition: %v", err)
|
|
|
|
}
|
|
|
|
}
|
2019-04-05 14:41:49 +00:00
|
|
|
|
2019-02-11 16:15:25 +00:00
|
|
|
// Fetch the epoch boundary root = hash_tree_root(epoch_boundary)
|
|
|
|
// where epoch_boundary is the block at the most recent epoch boundary in the
|
|
|
|
// chain defined by head -- i.e. the BeaconBlock where block.slot == get_epoch_start_slot(head.slot).
|
2019-02-25 02:09:45 +00:00
|
|
|
// If the epoch boundary slot is the same as state current slot,
|
|
|
|
// we set epoch boundary root to an empty root.
|
|
|
|
epochBoundaryRoot := make([]byte, 32)
|
2019-04-21 21:12:03 +00:00
|
|
|
epochStartSlot := helpers.StartSlot(helpers.SlotToEpoch(headState.Slot))
|
|
|
|
if epochStartSlot == headState.Slot {
|
2019-04-05 14:41:49 +00:00
|
|
|
epochBoundaryRoot = headRoot[:]
|
2019-02-25 02:09:45 +00:00
|
|
|
} else {
|
2019-04-05 14:41:49 +00:00
|
|
|
epochBoundaryRoot, err = blocks.BlockRoot(headState, epochStartSlot)
|
2019-02-25 02:09:45 +00:00
|
|
|
if err != nil {
|
2019-04-05 14:41:49 +00:00
|
|
|
return nil, fmt.Errorf("could not get epoch boundary block for slot %d: %v",
|
|
|
|
epochStartSlot, err)
|
2019-02-25 02:09:45 +00:00
|
|
|
}
|
2019-02-11 16:15:25 +00:00
|
|
|
}
|
2019-02-25 02:09:45 +00:00
|
|
|
// epoch_start_slot = get_epoch_start_slot(slot_to_epoch(head.slot))
|
2019-02-11 16:15:25 +00:00
|
|
|
// Fetch the justified block root = hash_tree_root(justified_block) where
|
|
|
|
// justified_block is the block at state.justified_epoch in the chain defined by head.
|
|
|
|
// On the server side, this is fetched by calling get_block_root(state, justified_epoch).
|
2019-02-25 02:09:45 +00:00
|
|
|
// If the last justified boundary slot is the same as state current slot (ex: slot 0),
|
|
|
|
// we set justified block root to an empty root.
|
2019-04-10 06:52:06 +00:00
|
|
|
justifiedBlockRoot := headState.JustifiedRoot
|
2019-02-25 02:09:45 +00:00
|
|
|
|
2019-04-10 06:52:06 +00:00
|
|
|
// If an attester has to attest for genesis block.
|
2019-04-05 14:41:49 +00:00
|
|
|
if headState.Slot == params.BeaconConfig().GenesisSlot {
|
2019-04-10 06:52:06 +00:00
|
|
|
epochBoundaryRoot = params.BeaconConfig().ZeroHash[:]
|
|
|
|
justifiedBlockRoot = params.BeaconConfig().ZeroHash[:]
|
2019-02-11 16:15:25 +00:00
|
|
|
}
|
2019-04-05 14:41:49 +00:00
|
|
|
|
2019-05-09 00:27:29 +00:00
|
|
|
res = &pb.AttestationDataResponse{
|
2019-04-05 14:41:49 +00:00
|
|
|
HeadSlot: headState.Slot,
|
|
|
|
BeaconBlockRootHash32: headRoot[:],
|
2019-02-25 02:09:45 +00:00
|
|
|
EpochBoundaryRootHash32: epochBoundaryRoot,
|
2019-04-05 14:41:49 +00:00
|
|
|
JustifiedEpoch: headState.JustifiedEpoch,
|
2019-02-25 02:09:45 +00:00
|
|
|
JustifiedBlockRootHash32: justifiedBlockRoot,
|
2019-04-05 14:41:49 +00:00
|
|
|
LatestCrosslink: headState.LatestCrosslinks[req.Shard],
|
2019-05-09 00:27:29 +00:00
|
|
|
}
|
|
|
|
if err := as.cache.Put(ctx, req, res); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return res, nil
|
2019-02-06 16:20:38 +00:00
|
|
|
}
|