mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-08 02:31:19 +00:00
3d8bb73f15
* initial validator attesthead rewrite based on proposer rewrite * proceed with fetching committees and tree hashing the canonical head at assigned attester slot * complete filling the properties of attestation data and all associated root hashes * add when to attest todo * finish entire attester client logic * tests with mocks checked in * tests passing in client * stubbed out server implementation * fixed build due to old property * regen mocks with new mockgen version * fixed broken tests * complete bazel build fix * address some review comments * deep proto test * server implementation logic * attestation info tests * completed passing tests, tree hash still blocked by error with nil fields unfortunately * fix todo failure * gazelle rerun and master merge * testing full behavior of attestation info server * finished all required server tests and implementation * break loop * fix tests due to genesis slot updates
78 lines
3.5 KiB
Go
78 lines
3.5 KiB
Go
package rpc
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/core/blocks"
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/core/helpers"
|
|
"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"
|
|
"github.com/prysmaticlabs/prysm/shared/hashutil"
|
|
)
|
|
|
|
// 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 {
|
|
beaconDB *db.BeaconDB
|
|
operationService operationService
|
|
}
|
|
|
|
// 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.
|
|
func (as *AttesterServer) AttestHead(ctx context.Context, att *pbp2p.Attestation) (*pb.AttestResponse, error) {
|
|
h, err := hashutil.HashProto(att)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("could not hash attestation: %v", err)
|
|
}
|
|
// Relays the attestation to chain service.
|
|
as.operationService.IncomingAttFeed().Send(att)
|
|
return &pb.AttestResponse{AttestationHash: h[:]}, nil
|
|
}
|
|
|
|
// AttestationInfoAtSlot fetches the necessary information from the current canonical head
|
|
// 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.
|
|
func (as *AttesterServer) AttestationInfoAtSlot(ctx context.Context, req *pb.AttestationInfoRequest) (*pb.AttestationInfoResponse, error) {
|
|
// 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.
|
|
head, err := as.beaconDB.BlockBySlot(req.Slot)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to retrieve chain head: %v", err)
|
|
}
|
|
if head == nil {
|
|
return nil, fmt.Errorf("no block found at slot %d", req.Slot)
|
|
}
|
|
blockRoot, err := hashutil.HashBeaconBlock(head) // TODO(#1461): Use Tree Hashing instead (IMPORTANT).
|
|
if err != nil {
|
|
return nil, fmt.Errorf("could not tree hash beacon block: %v", err)
|
|
}
|
|
beaconState, err := as.beaconDB.State()
|
|
if err != nil {
|
|
return nil, fmt.Errorf("could not fetch beacon state: %v", err)
|
|
}
|
|
// 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).
|
|
// On the server side, this is fetched by calling get_block_root(state, get_epoch_start_slot(head.slot)).
|
|
epochBoundaryRoot, err := blocks.BlockRoot(beaconState, helpers.StartSlot(head.Slot))
|
|
if err != nil {
|
|
return nil, fmt.Errorf("could not get epoch boundary block: %v", err)
|
|
}
|
|
// 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).
|
|
justifiedBlockRoot, err := blocks.BlockRoot(beaconState, beaconState.JustifiedEpoch)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("could not get justified block: %v", err)
|
|
}
|
|
return &pb.AttestationInfoResponse{
|
|
BeaconBlockRootHash32: blockRoot[:],
|
|
EpochBoundaryRootHash32: epochBoundaryRoot[:],
|
|
JustifiedEpoch: beaconState.JustifiedEpoch,
|
|
JustifiedBlockRootHash32: justifiedBlockRoot[:],
|
|
LatestCrosslink: beaconState.LatestCrosslinks[req.Shard],
|
|
}, nil
|
|
}
|