mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-19 00:04:12 +00:00
c1a9937760
* implemented all the merkle func signatures * branch indices complete * check for index out of range in generating merkle proof * completed full tests for sparse merkle trie impl * lint * formatting * gazelle * commentary * ivan comments * fmt * get rid of the deposit trie * recalculate trie when eth1 data changes * default data response recalculates tree * update merkle trie to use raw bytes * use the new verify merkle root func * builds * if default data response historical deposits are empty, return the deposit root at the contract at the time * work on trie * trying again with more logging * keep track of merkle trie indices, use correct big int ops * use uint for merkle idx * add todo * update ticker correctly * fix config and remove unnecessary logs * readd plus one fix * clarify some details * weird imports spacing * gazelle, lint * fix tests using the new deposit trie * builds but tests still fail * rpc tests * lint * tests pass * bazel lint * rem commented block * revert att slot fix * preston comments * comments * fix build * address last comment * use counter * imports
307 lines
12 KiB
Go
307 lines
12 KiB
Go
package rpc
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"math/big"
|
|
"time"
|
|
|
|
ptypes "github.com/gogo/protobuf/types"
|
|
"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/bytesutil"
|
|
"github.com/prysmaticlabs/prysm/shared/params"
|
|
"github.com/prysmaticlabs/prysm/shared/trieutil"
|
|
)
|
|
|
|
// BeaconServer defines a server implementation of the gRPC Beacon service,
|
|
// providing RPC endpoints for obtaining the canonical beacon chain head,
|
|
// fetching latest observed attestations, and more.
|
|
type BeaconServer struct {
|
|
beaconDB *db.BeaconDB
|
|
ctx context.Context
|
|
powChainService powChainService
|
|
chainService chainService
|
|
chainStartDelayFlag uint64
|
|
operationService operationService
|
|
incomingAttestation chan *pbp2p.Attestation
|
|
canonicalStateChan chan *pbp2p.BeaconState
|
|
chainStartChan chan time.Time
|
|
}
|
|
|
|
// WaitForChainStart queries the logs of the Deposit Contract in order to verify the beacon chain
|
|
// has started its runtime and validators begin their responsibilities. If it has not, it then
|
|
// subscribes to an event stream triggered by the powchain service whenever the ChainStart log does
|
|
// occur in the Deposit Contract on ETH 1.0.
|
|
func (bs *BeaconServer) WaitForChainStart(req *ptypes.Empty, stream pb.BeaconService_WaitForChainStartServer) error {
|
|
ok, genesisTime, err := bs.powChainService.HasChainStartLogOccurred()
|
|
if err != nil {
|
|
return fmt.Errorf("could not determine if ChainStart log has occurred: %v", err)
|
|
}
|
|
if ok && bs.chainStartDelayFlag == 0 {
|
|
res := &pb.ChainStartResponse{
|
|
Started: true,
|
|
GenesisTime: genesisTime,
|
|
}
|
|
return stream.Send(res)
|
|
}
|
|
|
|
sub := bs.chainService.StateInitializedFeed().Subscribe(bs.chainStartChan)
|
|
defer sub.Unsubscribe()
|
|
for {
|
|
select {
|
|
case chainStartTime := <-bs.chainStartChan:
|
|
log.Info("Sending ChainStart log and genesis time to connected validator clients")
|
|
res := &pb.ChainStartResponse{
|
|
Started: true,
|
|
GenesisTime: uint64(chainStartTime.Unix()),
|
|
}
|
|
return stream.Send(res)
|
|
case <-sub.Err():
|
|
return errors.New("subscriber closed, exiting goroutine")
|
|
case <-bs.ctx.Done():
|
|
return errors.New("rpc context closed, exiting goroutine")
|
|
}
|
|
}
|
|
}
|
|
|
|
// CanonicalHead of the current beacon chain. This method is requested on-demand
|
|
// by a validator when it is their time to propose or attest.
|
|
func (bs *BeaconServer) CanonicalHead(ctx context.Context, req *ptypes.Empty) (*pbp2p.BeaconBlock, error) {
|
|
block, err := bs.beaconDB.ChainHead()
|
|
if err != nil {
|
|
return nil, fmt.Errorf("could not get canonical head block: %v", err)
|
|
}
|
|
return block, nil
|
|
}
|
|
|
|
// LatestAttestation streams the latest processed attestations to the rpc clients.
|
|
func (bs *BeaconServer) LatestAttestation(req *ptypes.Empty, stream pb.BeaconService_LatestAttestationServer) error {
|
|
sub := bs.operationService.IncomingAttFeed().Subscribe(bs.incomingAttestation)
|
|
defer sub.Unsubscribe()
|
|
for {
|
|
select {
|
|
case attestation := <-bs.incomingAttestation:
|
|
log.Info("Sending attestation to RPC clients")
|
|
if err := stream.Send(attestation); err != nil {
|
|
return err
|
|
}
|
|
case <-sub.Err():
|
|
log.Debug("Subscriber closed, exiting goroutine")
|
|
return nil
|
|
case <-bs.ctx.Done():
|
|
log.Debug("RPC context closed, exiting goroutine")
|
|
return nil
|
|
}
|
|
}
|
|
}
|
|
|
|
// ForkData fetches the current fork information from the beacon state.
|
|
func (bs *BeaconServer) ForkData(ctx context.Context, _ *ptypes.Empty) (*pbp2p.Fork, error) {
|
|
state, err := bs.beaconDB.State(ctx)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("could not retrieve beacon state: %v", err)
|
|
}
|
|
return state.Fork, nil
|
|
}
|
|
|
|
// Eth1Data is a mechanism used by block proposers vote on a recent Ethereum 1.0 block hash and an
|
|
// associated deposit root found in the Ethereum 1.0 deposit contract. When consensus is formed,
|
|
// state.latest_eth1_data is updated, and validator deposits up to this root can be processed.
|
|
// The deposit root can be calculated by calling the get_deposit_root() function of
|
|
// the deposit contract using the post-state of the block hash.
|
|
func (bs *BeaconServer) Eth1Data(ctx context.Context, _ *ptypes.Empty) (*pb.Eth1DataResponse, error) {
|
|
beaconState, err := bs.beaconDB.State(ctx)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("could not fetch beacon state: %v", err)
|
|
}
|
|
// Fetch the current canonical chain height from the eth1.0 chain.
|
|
currentHeight := bs.powChainService.LatestBlockHeight()
|
|
eth1FollowDistance := int64(params.BeaconConfig().Eth1FollowDistance)
|
|
|
|
stateLatestEth1Hash := bytesutil.ToBytes32(beaconState.LatestEth1Data.BlockHash32)
|
|
// If latest ETH1 block hash is empty, send a default response
|
|
if stateLatestEth1Hash == [32]byte{} {
|
|
return bs.defaultDataResponse(ctx, currentHeight, eth1FollowDistance)
|
|
}
|
|
// Fetch the height of the block pointed to by the beacon state's latest_eth1_data.block_hash
|
|
// in the canonical, eth1.0 chain.
|
|
_, stateLatestEth1Height, err := bs.powChainService.BlockExists(ctx, stateLatestEth1Hash)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("could not verify block with hash exists in Eth1 chain: %#x: %v", stateLatestEth1Hash, err)
|
|
}
|
|
dataVotes := []*pbp2p.Eth1DataVote{}
|
|
bestVote := &pbp2p.Eth1DataVote{}
|
|
bestVoteHeight := big.NewInt(0)
|
|
for _, vote := range beaconState.Eth1DataVotes {
|
|
eth1Hash := bytesutil.ToBytes32(vote.Eth1Data.BlockHash32)
|
|
// Verify the block from the vote's block hash exists in the eth1.0 chain and fetch its height.
|
|
blockExists, blockHeight, err := bs.powChainService.BlockExists(ctx, eth1Hash)
|
|
if err != nil {
|
|
log.Debugf("Could not verify block with hash exists in Eth1 chain: %#x: %v", eth1Hash, err)
|
|
continue
|
|
}
|
|
if !blockExists {
|
|
continue
|
|
}
|
|
// Let dataVotes be the set of Eth1DataVote objects vote in state.eth1_data_votes where:
|
|
// vote.eth1_data.block_hash is the hash of an eth1.0 block that is:
|
|
// (i) part of the canonical chain
|
|
// (ii) >= ETH1_FOLLOW_DISTANCE blocks behind the head
|
|
// (iii) newer than state.latest_eth1_data.block_data.
|
|
// vote.eth1_data.deposit_root is the deposit root of the eth1.0 deposit contract
|
|
// at the block defined by vote.eth1_data.block_hash.
|
|
isBehindFollowDistance := big.NewInt(0).Sub(currentHeight, big.NewInt(eth1FollowDistance)).Cmp(blockHeight) >= 0
|
|
isAheadStateLatestEth1Data := blockHeight.Cmp(stateLatestEth1Height) == 1
|
|
if blockExists && isBehindFollowDistance && isAheadStateLatestEth1Data {
|
|
dataVotes = append(dataVotes, vote)
|
|
|
|
// Sets the first vote as best vote.
|
|
if len(dataVotes) == 1 {
|
|
bestVote = vote
|
|
bestVoteHeight = blockHeight
|
|
continue
|
|
}
|
|
// If dataVotes is non-empty:
|
|
// Let best_vote be the member of D that has the highest vote.eth1_data.vote_count,
|
|
// breaking ties by favoring block hashes with higher associated block height.
|
|
// Let block_hash = best_vote.eth1_data.block_hash.
|
|
// Let deposit_root = best_vote.eth1_data.deposit_root.
|
|
if vote.VoteCount > bestVote.VoteCount {
|
|
bestVote = vote
|
|
bestVoteHeight = blockHeight
|
|
} else if vote.VoteCount == bestVote.VoteCount {
|
|
|
|
if blockHeight.Cmp(bestVoteHeight) == 1 {
|
|
bestVote = vote
|
|
bestVoteHeight = blockHeight
|
|
}
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
// Now we handle the following two scenarios:
|
|
// If dataVotes is empty:
|
|
// Let block_hash be the block hash of the ETH1_FOLLOW_DISTANCE'th ancestor of the head of
|
|
// the canonical eth1.0 chain.
|
|
// Let deposit_root be the deposit root of the eth1.0 deposit contract in the
|
|
// post-state of the block referenced by block_hash.
|
|
if len(dataVotes) == 0 {
|
|
return bs.defaultDataResponse(ctx, currentHeight, eth1FollowDistance)
|
|
}
|
|
|
|
return &pb.Eth1DataResponse{
|
|
Eth1Data: &pbp2p.Eth1Data{
|
|
BlockHash32: bestVote.Eth1Data.BlockHash32,
|
|
DepositRootHash32: bestVote.Eth1Data.DepositRootHash32,
|
|
},
|
|
}, nil
|
|
}
|
|
|
|
// PendingDeposits returns a list of pending deposits that are ready for
|
|
// inclusion in the next beacon block.
|
|
func (bs *BeaconServer) PendingDeposits(ctx context.Context, _ *ptypes.Empty) (*pb.PendingDepositsResponse, error) {
|
|
bNum := bs.powChainService.LatestBlockHeight()
|
|
if bNum == nil {
|
|
return nil, errors.New("latest PoW block number is unknown")
|
|
}
|
|
// Only request deposits that have passed the ETH1 follow distance window.
|
|
bNum = bNum.Sub(bNum, big.NewInt(int64(params.BeaconConfig().Eth1FollowDistance)))
|
|
allDeps := bs.beaconDB.AllDeposits(ctx, bNum)
|
|
if len(allDeps) == 0 {
|
|
return &pb.PendingDepositsResponse{PendingDeposits: nil}, nil
|
|
}
|
|
|
|
pendingDeps := bs.beaconDB.PendingDeposits(ctx, bNum)
|
|
// Need to fetch if the deposits up to the state's latest eth 1 data matches
|
|
// the number of all deposits in this RPC call. If not, then we return nil.
|
|
beaconState, err := bs.beaconDB.State(ctx)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("could not fetch beacon state: %v", err)
|
|
}
|
|
h := bytesutil.ToBytes32(beaconState.LatestEth1Data.BlockHash32)
|
|
_, latestEth1DataHeight, err := bs.powChainService.BlockExists(ctx, h)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("could not fetch eth1data height: %v", err)
|
|
}
|
|
// If the state's latest eth1 data's block hash has a height of 100, we fetch all the deposits up to height 100.
|
|
// If this doesn't match the number of deposits stored in the cache, the generated trie will not be the same and
|
|
// root will fail to verify. This can happen in a scenario where we perhaps have a deposit from height 101,
|
|
// so we want to avoid any possible mismatches in these lengths.
|
|
upToLatestEth1DataDeposits := bs.beaconDB.AllDeposits(ctx, latestEth1DataHeight)
|
|
if len(upToLatestEth1DataDeposits) != len(allDeps) {
|
|
return &pb.PendingDepositsResponse{PendingDeposits: nil}, nil
|
|
}
|
|
depositData := [][]byte{}
|
|
indices := []uint64{}
|
|
for i := range upToLatestEth1DataDeposits {
|
|
depositData = append(depositData, upToLatestEth1DataDeposits[i].DepositData)
|
|
indices = append(indices, upToLatestEth1DataDeposits[i].MerkleTreeIndex)
|
|
}
|
|
|
|
depositTrie, err := trieutil.GenerateTrieFromItems(depositData, int(params.BeaconConfig().DepositContractTreeDepth))
|
|
if err != nil {
|
|
return nil, fmt.Errorf("could not generate historical deposit trie from deposits: %v", err)
|
|
}
|
|
for i := range pendingDeps {
|
|
pendingDeps[i], err = constructMerkleProof(depositTrie, pendingDeps[i])
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
return &pb.PendingDepositsResponse{PendingDeposits: pendingDeps}, nil
|
|
}
|
|
|
|
func (bs *BeaconServer) defaultDataResponse(ctx context.Context, currentHeight *big.Int, eth1FollowDistance int64) (*pb.Eth1DataResponse, error) {
|
|
ancestorHeight := big.NewInt(0).Sub(currentHeight, big.NewInt(eth1FollowDistance))
|
|
blockHash, err := bs.powChainService.BlockHashByHeight(ctx, ancestorHeight)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("could not fetch ETH1_FOLLOW_DISTANCE ancestor: %v", err)
|
|
}
|
|
// Fetch all historical deposits up to an ancestor height.
|
|
allDeposits := bs.beaconDB.AllDeposits(ctx, ancestorHeight)
|
|
depositData := [][]byte{}
|
|
// If there are less than or equal to len(ChainStartDeposits) historical deposits, then we just fetch the default
|
|
// deposit root obtained from constructing the Merkle trie with the ChainStart deposits.
|
|
chainStartDeposits := bs.powChainService.ChainStartDeposits()
|
|
if len(allDeposits) <= len(chainStartDeposits) {
|
|
depositData = chainStartDeposits
|
|
} else {
|
|
indices := []uint64{}
|
|
for i := range allDeposits {
|
|
depositData = append(depositData, allDeposits[i].DepositData)
|
|
indices = append(indices, allDeposits[i].MerkleTreeIndex)
|
|
}
|
|
}
|
|
depositTrie, err := trieutil.GenerateTrieFromItems(depositData, int(params.BeaconConfig().DepositContractTreeDepth))
|
|
if err != nil {
|
|
return nil, fmt.Errorf("could not generate historical deposit trie from deposits: %v", err)
|
|
}
|
|
depositRoot := depositTrie.Root()
|
|
return &pb.Eth1DataResponse{
|
|
Eth1Data: &pbp2p.Eth1Data{
|
|
DepositRootHash32: depositRoot[:],
|
|
BlockHash32: blockHash[:],
|
|
},
|
|
}, nil
|
|
}
|
|
|
|
func constructMerkleProof(trie *trieutil.MerkleTrie, deposit *pbp2p.Deposit) (*pbp2p.Deposit, error) {
|
|
proof, err := trie.MerkleProof(int(deposit.MerkleTreeIndex))
|
|
if err != nil {
|
|
return nil, fmt.Errorf(
|
|
"could not generate merkle proof for deposit at index %d: %v",
|
|
deposit.MerkleTreeIndex,
|
|
err,
|
|
)
|
|
}
|
|
// For every deposit, we construct a Merkle proof using the powchain service's
|
|
// in-memory deposits trie, which is updated only once the state's LatestETH1Data
|
|
// property changes during a state transition after a voting period.
|
|
deposit.MerkleBranchHash32S = proof
|
|
return deposit, nil
|
|
}
|