mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-04 08:44:28 +00:00
de6ea81790
* add metrics for corret participation rate * gaz * add recent canonical roots API * gaz * all tests passed * comments * fmt
278 lines
9.7 KiB
Go
278 lines
9.7 KiB
Go
// Package blockchain defines the life-cycle and status of the beacon chain
|
|
// as well as the Ethereum Serenity beacon chain fork-choice rule based on
|
|
// Casper Proof of Stake finality.
|
|
package blockchain
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"fmt"
|
|
"sort"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/attestation"
|
|
b "github.com/prysmaticlabs/prysm/beacon-chain/core/blocks"
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/db"
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/operations"
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/powchain"
|
|
pb "github.com/prysmaticlabs/prysm/proto/beacon/p2p/v1"
|
|
pbrpc "github.com/prysmaticlabs/prysm/proto/beacon/rpc/v1"
|
|
"github.com/prysmaticlabs/prysm/shared/event"
|
|
"github.com/prysmaticlabs/prysm/shared/hashutil"
|
|
"github.com/prysmaticlabs/prysm/shared/p2p"
|
|
"github.com/sirupsen/logrus"
|
|
"go.opencensus.io/trace"
|
|
)
|
|
|
|
var log = logrus.WithField("prefix", "blockchain")
|
|
|
|
// ChainFeeds interface defines the methods of the ChainService which provide
|
|
// information feeds.
|
|
type ChainFeeds interface {
|
|
StateInitializedFeed() *event.Feed
|
|
}
|
|
|
|
// ChainService represents a service that handles the internal
|
|
// logic of managing the full PoS beacon chain.
|
|
type ChainService struct {
|
|
ctx context.Context
|
|
cancel context.CancelFunc
|
|
beaconDB *db.BeaconDB
|
|
web3Service *powchain.Web3Service
|
|
attsService attestation.TargetHandler
|
|
opsPoolService operations.OperationFeeds
|
|
chainStartChan chan time.Time
|
|
canonicalBlockFeed *event.Feed
|
|
genesisTime time.Time
|
|
finalizedEpoch uint64
|
|
stateInitializedFeed *event.Feed
|
|
p2p p2p.Broadcaster
|
|
canonicalBlocks map[uint64][]byte
|
|
canonicalBlocksLock sync.RWMutex
|
|
receiveBlockLock sync.Mutex
|
|
}
|
|
|
|
// Config options for the service.
|
|
type Config struct {
|
|
BeaconBlockBuf int
|
|
Web3Service *powchain.Web3Service
|
|
AttsService attestation.TargetHandler
|
|
BeaconDB *db.BeaconDB
|
|
OpsPoolService operations.OperationFeeds
|
|
DevMode bool
|
|
P2p p2p.Broadcaster
|
|
}
|
|
|
|
// NewChainService instantiates a new service instance that will
|
|
// be registered into a running beacon node.
|
|
func NewChainService(ctx context.Context, cfg *Config) (*ChainService, error) {
|
|
ctx, cancel := context.WithCancel(ctx)
|
|
return &ChainService{
|
|
ctx: ctx,
|
|
cancel: cancel,
|
|
beaconDB: cfg.BeaconDB,
|
|
web3Service: cfg.Web3Service,
|
|
opsPoolService: cfg.OpsPoolService,
|
|
attsService: cfg.AttsService,
|
|
canonicalBlockFeed: new(event.Feed),
|
|
chainStartChan: make(chan time.Time),
|
|
stateInitializedFeed: new(event.Feed),
|
|
p2p: cfg.P2p,
|
|
canonicalBlocks: make(map[uint64][]byte),
|
|
}, nil
|
|
}
|
|
|
|
// Start a blockchain service's main event loop.
|
|
func (c *ChainService) Start() {
|
|
beaconState, err := c.beaconDB.HeadState(c.ctx)
|
|
if err != nil {
|
|
log.Fatalf("Could not fetch beacon state: %v", err)
|
|
}
|
|
// If the chain has already been initialized, simply start the block processing routine.
|
|
if beaconState != nil {
|
|
log.Info("Beacon chain data already exists, starting service")
|
|
c.genesisTime = time.Unix(int64(beaconState.GenesisTime), 0)
|
|
c.finalizedEpoch = beaconState.FinalizedEpoch
|
|
} else {
|
|
log.Info("Waiting for ChainStart log from the Validator Deposit Contract to start the beacon chain...")
|
|
if c.web3Service == nil {
|
|
log.Fatal("Not configured web3Service for POW chain")
|
|
return // return need for TestStartUninitializedChainWithoutConfigPOWChain.
|
|
}
|
|
subChainStart := c.web3Service.ChainStartFeed().Subscribe(c.chainStartChan)
|
|
go func() {
|
|
genesisTime := <-c.chainStartChan
|
|
c.processChainStartTime(genesisTime, subChainStart)
|
|
return
|
|
}()
|
|
}
|
|
}
|
|
|
|
// processChainStartTime initializes a series of deposits from the ChainStart deposits in the eth1
|
|
// deposit contract, initializes the beacon chain's state, and kicks off the beacon chain.
|
|
func (c *ChainService) processChainStartTime(genesisTime time.Time, chainStartSub event.Subscription) {
|
|
initialDepositsData := c.web3Service.ChainStartDeposits()
|
|
initialDeposits := make([]*pb.Deposit, len(initialDepositsData))
|
|
for i := range initialDepositsData {
|
|
initialDeposits[i] = &pb.Deposit{DepositData: initialDepositsData[i]}
|
|
}
|
|
|
|
beaconState, err := c.initializeBeaconChain(genesisTime, initialDeposits, c.web3Service.ChainStartETH1Data())
|
|
if err != nil {
|
|
log.Fatalf("Could not initialize beacon chain: %v", err)
|
|
}
|
|
c.finalizedEpoch = beaconState.FinalizedEpoch
|
|
c.stateInitializedFeed.Send(genesisTime)
|
|
chainStartSub.Unsubscribe()
|
|
}
|
|
|
|
// initializes the state and genesis block of the beacon chain to persistent storage
|
|
// based on a genesis timestamp value obtained from the ChainStart event emitted
|
|
// by the ETH1.0 Deposit Contract and the POWChain service of the node.
|
|
func (c *ChainService) initializeBeaconChain(genesisTime time.Time, deposits []*pb.Deposit,
|
|
eth1data *pb.Eth1Data) (*pb.BeaconState, error) {
|
|
ctx, span := trace.StartSpan(context.Background(), "beacon-chain.ChainService.initializeBeaconChain")
|
|
defer span.End()
|
|
log.Info("ChainStart time reached, starting the beacon chain!")
|
|
c.genesisTime = genesisTime
|
|
unixTime := uint64(genesisTime.Unix())
|
|
if err := c.beaconDB.InitializeState(c.ctx, unixTime, deposits, eth1data); err != nil {
|
|
return nil, fmt.Errorf("could not initialize beacon state to disk: %v", err)
|
|
}
|
|
beaconState, err := c.beaconDB.HeadState(c.ctx)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("could not attempt fetch beacon state: %v", err)
|
|
}
|
|
|
|
stateRoot, err := hashutil.HashProto(beaconState)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("could not hash beacon state: %v", err)
|
|
}
|
|
genBlock := b.NewGenesisBlock(stateRoot[:])
|
|
genBlockRoot, err := hashutil.HashBeaconBlock(genBlock)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("could not hash beacon block: %v", err)
|
|
}
|
|
|
|
// TODO(#2011): Remove this in state caching.
|
|
beaconState.LatestBlock = genBlock
|
|
|
|
if err := c.beaconDB.SaveBlock(genBlock); err != nil {
|
|
return nil, fmt.Errorf("could not save genesis block to disk: %v", err)
|
|
}
|
|
if err := c.beaconDB.SaveAttestationTarget(ctx, &pb.AttestationTarget{
|
|
Slot: genBlock.Slot,
|
|
BlockRoot: genBlockRoot[:],
|
|
ParentRoot: genBlock.ParentRootHash32,
|
|
}); err != nil {
|
|
return nil, fmt.Errorf("failed to save attestation target: %v", err)
|
|
}
|
|
if err := c.beaconDB.UpdateChainHead(ctx, genBlock, beaconState); err != nil {
|
|
return nil, fmt.Errorf("could not set chain head, %v", err)
|
|
}
|
|
if err := c.beaconDB.SaveJustifiedBlock(genBlock); err != nil {
|
|
return nil, fmt.Errorf("could not save gensis block as justified block: %v", err)
|
|
}
|
|
if err := c.beaconDB.SaveFinalizedBlock(genBlock); err != nil {
|
|
return nil, fmt.Errorf("could not save gensis block as finalized block: %v", err)
|
|
}
|
|
if err := c.beaconDB.SaveJustifiedState(beaconState); err != nil {
|
|
return nil, fmt.Errorf("could not save gensis state as justified state: %v", err)
|
|
}
|
|
if err := c.beaconDB.SaveFinalizedState(beaconState); err != nil {
|
|
return nil, fmt.Errorf("could not save gensis state as finalized state: %v", err)
|
|
}
|
|
return beaconState, nil
|
|
}
|
|
|
|
// Stop the blockchain service's main event loop and associated goroutines.
|
|
func (c *ChainService) Stop() error {
|
|
defer c.cancel()
|
|
|
|
log.Info("Stopping service")
|
|
return nil
|
|
}
|
|
|
|
// Status always returns nil.
|
|
// TODO(1202): Add service health checks.
|
|
func (c *ChainService) Status() error {
|
|
return nil
|
|
}
|
|
|
|
// CanonicalBlockFeed returns a channel that is written to
|
|
// whenever a new block is determined to be canonical in the chain.
|
|
func (c *ChainService) CanonicalBlockFeed() *event.Feed {
|
|
return c.canonicalBlockFeed
|
|
}
|
|
|
|
// StateInitializedFeed returns a feed that is written to
|
|
// when the beacon state is first initialized.
|
|
func (c *ChainService) StateInitializedFeed() *event.Feed {
|
|
return c.stateInitializedFeed
|
|
}
|
|
|
|
// ChainHeadRoot returns the hash root of the last beacon block processed by the
|
|
// block chain service.
|
|
func (c *ChainService) ChainHeadRoot() ([32]byte, error) {
|
|
head, err := c.beaconDB.ChainHead()
|
|
if err != nil {
|
|
return [32]byte{}, fmt.Errorf("could not retrieve chain head: %v", err)
|
|
}
|
|
|
|
root, err := hashutil.HashBeaconBlock(head)
|
|
if err != nil {
|
|
return [32]byte{}, fmt.Errorf("could not tree hash parent block: %v", err)
|
|
}
|
|
return root, nil
|
|
}
|
|
|
|
// IsCanonical returns true if the input block hash of the corresponding slot
|
|
// is part of the canonical chain. False otherwise.
|
|
func (c *ChainService) IsCanonical(slot uint64, hash []byte) bool {
|
|
c.canonicalBlocksLock.RLock()
|
|
defer c.canonicalBlocksLock.RUnlock()
|
|
if canonicalHash, ok := c.canonicalBlocks[slot]; ok {
|
|
return bytes.Equal(canonicalHash, hash)
|
|
}
|
|
return false
|
|
}
|
|
|
|
// RecentCanonicalRoots returns the latest block slot and root of the canonical block chain,
|
|
// the block slots and roots are sorted and in descending order. Input count determines
|
|
// the number of block slots and roots to return.
|
|
func (c *ChainService) RecentCanonicalRoots(count uint64) []*pbrpc.BlockRoot {
|
|
c.canonicalBlocksLock.RLock()
|
|
defer c.canonicalBlocksLock.RUnlock()
|
|
var slots []int
|
|
for s := range c.canonicalBlocks {
|
|
slots = append(slots, int(s))
|
|
}
|
|
|
|
// Return the all the canonical blocks if the input count is greater than
|
|
// the depth of the block tree.
|
|
totalRoots := uint64(len(slots))
|
|
if count > totalRoots {
|
|
count = totalRoots
|
|
}
|
|
|
|
sort.Sort(sort.Reverse(sort.IntSlice(slots)))
|
|
blockRoots := make([]*pbrpc.BlockRoot, count)
|
|
for i := 0; i < int(count); i++ {
|
|
slot := uint64(slots[i])
|
|
blockRoots[i] = &pbrpc.BlockRoot{
|
|
Slot: slot,
|
|
Root: c.canonicalBlocks[slot],
|
|
}
|
|
}
|
|
return blockRoots
|
|
}
|
|
|
|
// InsertsCanonical inserts a canonical block hash to its corresponding slot.
|
|
// This is used for testing purpose.
|
|
func (c *ChainService) InsertsCanonical(slot uint64, hash []byte) {
|
|
c.canonicalBlocksLock.Lock()
|
|
defer c.canonicalBlocksLock.Unlock()
|
|
c.canonicalBlocks[slot] = hash
|
|
}
|