2019-01-19 02:57:51 +00:00
|
|
|
// 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.
|
2018-07-19 16:31:50 +00:00
|
|
|
package blockchain
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2018-07-31 04:41:27 +00:00
|
|
|
"fmt"
|
2019-05-15 14:38:27 +00:00
|
|
|
"runtime"
|
2019-04-16 18:19:31 +00:00
|
|
|
"sync"
|
2018-09-18 13:06:28 +00:00
|
|
|
"time"
|
2018-07-19 16:31:50 +00:00
|
|
|
|
2019-08-02 02:27:38 +00:00
|
|
|
"github.com/pkg/errors"
|
2019-11-27 05:08:18 +00:00
|
|
|
ethpb "github.com/prysmaticlabs/ethereumapis/eth/v1alpha1"
|
2019-10-01 20:05:17 +00:00
|
|
|
"github.com/prysmaticlabs/go-ssz"
|
2019-08-21 17:40:00 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/blockchain/forkchoice"
|
2019-08-21 16:04:00 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/cache/depositcache"
|
2019-08-24 04:02:34 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/core/blocks"
|
2019-12-07 17:57:26 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/core/feed"
|
|
|
|
statefeed "github.com/prysmaticlabs/prysm/beacon-chain/core/feed/state"
|
2019-09-27 22:56:08 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/core/helpers"
|
2019-08-22 01:14:24 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/core/state"
|
2018-11-16 17:01:41 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/db"
|
2019-03-19 23:07:49 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/operations"
|
2019-08-22 01:14:24 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/p2p"
|
2018-07-22 16:58:14 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/powchain"
|
2019-08-23 01:13:56 +00:00
|
|
|
pb "github.com/prysmaticlabs/prysm/proto/beacon/p2p/v1"
|
|
|
|
"github.com/prysmaticlabs/prysm/shared/bytesutil"
|
2019-09-27 22:56:08 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/featureconfig"
|
2018-07-21 17:51:18 +00:00
|
|
|
"github.com/sirupsen/logrus"
|
2019-04-02 08:49:45 +00:00
|
|
|
"go.opencensus.io/trace"
|
2018-07-19 16:31:50 +00:00
|
|
|
)
|
|
|
|
|
2019-09-07 02:39:14 +00:00
|
|
|
// Service represents a service that handles the internal
|
2018-07-19 16:31:50 +00:00
|
|
|
// logic of managing the full PoS beacon chain.
|
2019-09-07 02:39:14 +00:00
|
|
|
type Service struct {
|
2019-11-23 03:35:47 +00:00
|
|
|
ctx context.Context
|
|
|
|
cancel context.CancelFunc
|
|
|
|
beaconDB db.Database
|
|
|
|
depositCache *depositcache.DepositCache
|
|
|
|
chainStartFetcher powchain.ChainStartFetcher
|
|
|
|
opsPoolService operations.OperationFeeds
|
|
|
|
forkChoiceStore forkchoice.ForkChoicer
|
|
|
|
genesisTime time.Time
|
|
|
|
p2p p2p.Broadcaster
|
|
|
|
maxRoutines int64
|
|
|
|
headSlot uint64
|
|
|
|
headBlock *ethpb.BeaconBlock
|
|
|
|
headState *pb.BeaconState
|
|
|
|
canonicalRoots map[uint64][]byte
|
|
|
|
headLock sync.RWMutex
|
|
|
|
stateNotifier statefeed.Notifier
|
2018-07-19 16:31:50 +00:00
|
|
|
}
|
|
|
|
|
2018-08-09 22:54:59 +00:00
|
|
|
// Config options for the service.
|
|
|
|
type Config struct {
|
2019-09-09 21:13:50 +00:00
|
|
|
BeaconBlockBuf int
|
|
|
|
ChainStartFetcher powchain.ChainStartFetcher
|
|
|
|
BeaconDB db.Database
|
|
|
|
DepositCache *depositcache.DepositCache
|
|
|
|
OpsPoolService operations.OperationFeeds
|
|
|
|
P2p p2p.Broadcaster
|
|
|
|
MaxRoutines int64
|
2019-11-19 22:15:48 +00:00
|
|
|
StateNotifier statefeed.Notifier
|
2018-08-09 22:54:59 +00:00
|
|
|
}
|
|
|
|
|
2019-09-07 02:39:14 +00:00
|
|
|
// NewService instantiates a new block service instance that will
|
2018-07-19 16:31:50 +00:00
|
|
|
// be registered into a running beacon node.
|
2019-09-07 02:39:14 +00:00
|
|
|
func NewService(ctx context.Context, cfg *Config) (*Service, error) {
|
2018-07-19 16:31:50 +00:00
|
|
|
ctx, cancel := context.WithCancel(ctx)
|
2019-08-21 17:40:00 +00:00
|
|
|
store := forkchoice.NewForkChoiceService(ctx, cfg.BeaconDB)
|
2019-09-07 02:39:14 +00:00
|
|
|
return &Service{
|
2019-11-23 03:35:47 +00:00
|
|
|
ctx: ctx,
|
|
|
|
cancel: cancel,
|
|
|
|
beaconDB: cfg.BeaconDB,
|
|
|
|
depositCache: cfg.DepositCache,
|
|
|
|
chainStartFetcher: cfg.ChainStartFetcher,
|
|
|
|
opsPoolService: cfg.OpsPoolService,
|
|
|
|
forkChoiceStore: store,
|
|
|
|
p2p: cfg.P2p,
|
|
|
|
canonicalRoots: make(map[uint64][]byte),
|
|
|
|
maxRoutines: cfg.MaxRoutines,
|
|
|
|
stateNotifier: cfg.StateNotifier,
|
2018-07-31 04:41:27 +00:00
|
|
|
}, nil
|
2018-07-19 16:31:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Start a blockchain service's main event loop.
|
2019-09-07 02:39:14 +00:00
|
|
|
func (s *Service) Start() {
|
2019-09-03 18:06:35 +00:00
|
|
|
ctx := context.TODO()
|
2019-09-07 02:39:14 +00:00
|
|
|
beaconState, err := s.beaconDB.HeadState(ctx)
|
2018-10-02 20:07:33 +00:00
|
|
|
if err != nil {
|
2019-01-28 11:59:37 +00:00
|
|
|
log.Fatalf("Could not fetch beacon state: %v", err)
|
2018-10-02 20:07:33 +00:00
|
|
|
}
|
2019-12-06 00:49:19 +00:00
|
|
|
|
|
|
|
// For running initial sync with state cache, in an event of restart, we use
|
|
|
|
// last finalized check point as start point to sync instead of head
|
|
|
|
// state. This is because we no longer save state every slot during sync.
|
|
|
|
if featureconfig.Get().InitSyncCacheState {
|
|
|
|
cp, err := s.beaconDB.FinalizedCheckpoint(ctx)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf("Could not fetch finalized cp: %v", err)
|
|
|
|
}
|
|
|
|
if beaconState == nil {
|
|
|
|
beaconState, err = s.beaconDB.State(ctx, bytesutil.ToBytes32(cp.Root))
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf("Could not fetch beacon state: %v", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-28 11:59:37 +00:00
|
|
|
// If the chain has already been initialized, simply start the block processing routine.
|
|
|
|
if beaconState != nil {
|
2019-10-01 20:05:17 +00:00
|
|
|
log.Info("Blockchain data already exists in DB, initializing...")
|
2019-09-07 02:39:14 +00:00
|
|
|
s.genesisTime = time.Unix(int64(beaconState.GenesisTime), 0)
|
|
|
|
if err := s.initializeChainInfo(ctx); err != nil {
|
2019-08-30 22:24:37 +00:00
|
|
|
log.Fatalf("Could not set up chain info: %v", err)
|
|
|
|
}
|
2019-09-07 02:39:14 +00:00
|
|
|
justifiedCheckpoint, err := s.beaconDB.JustifiedCheckpoint(ctx)
|
2019-08-29 22:32:35 +00:00
|
|
|
if err != nil {
|
2019-08-30 13:58:02 +00:00
|
|
|
log.Fatalf("Could not get justified checkpoint: %v", err)
|
2019-08-29 22:32:35 +00:00
|
|
|
}
|
2019-09-07 02:39:14 +00:00
|
|
|
finalizedCheckpoint, err := s.beaconDB.FinalizedCheckpoint(ctx)
|
2019-08-30 13:58:02 +00:00
|
|
|
if err != nil {
|
|
|
|
log.Fatalf("Could not get finalized checkpoint: %v", err)
|
|
|
|
}
|
2019-09-07 02:39:14 +00:00
|
|
|
if err := s.forkChoiceStore.GenesisStore(ctx, justifiedCheckpoint, finalizedCheckpoint); err != nil {
|
2019-08-29 22:32:35 +00:00
|
|
|
log.Fatalf("Could not start fork choice service: %v", err)
|
|
|
|
}
|
2019-12-07 17:57:26 +00:00
|
|
|
s.stateNotifier.StateFeed().Send(&feed.Event{
|
|
|
|
Type: statefeed.Initialized,
|
|
|
|
Data: &statefeed.InitializedData{
|
2019-11-23 03:35:47 +00:00
|
|
|
StartTime: s.genesisTime,
|
|
|
|
},
|
|
|
|
})
|
2019-01-28 11:59:37 +00:00
|
|
|
} else {
|
2019-10-01 20:05:17 +00:00
|
|
|
log.Info("Waiting to reach the validator deposit threshold to start the beacon chain...")
|
2019-09-09 21:13:50 +00:00
|
|
|
if s.chainStartFetcher == nil {
|
2019-02-07 16:11:18 +00:00
|
|
|
log.Fatal("Not configured web3Service for POW chain")
|
2019-02-14 20:04:47 +00:00
|
|
|
return // return need for TestStartUninitializedChainWithoutConfigPOWChain.
|
2019-02-07 16:11:18 +00:00
|
|
|
}
|
2019-01-28 11:59:37 +00:00
|
|
|
go func() {
|
2019-12-07 17:57:26 +00:00
|
|
|
stateChannel := make(chan *feed.Event, 1)
|
2019-11-23 03:35:47 +00:00
|
|
|
stateSub := s.stateNotifier.StateFeed().Subscribe(stateChannel)
|
|
|
|
defer stateSub.Unsubscribe()
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case event := <-stateChannel:
|
|
|
|
if event.Type == statefeed.ChainStarted {
|
|
|
|
data := event.Data.(*statefeed.ChainStartedData)
|
|
|
|
log.WithField("starttime", data.StartTime).Debug("Received chain start event")
|
|
|
|
s.processChainStartTime(ctx, data.StartTime)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
case <-s.ctx.Done():
|
|
|
|
log.Debug("Context closed, exiting goroutine")
|
|
|
|
return
|
|
|
|
case err := <-stateSub.Err():
|
|
|
|
log.WithError(err).Error("Subscription to state notifier failed")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
2019-01-28 11:59:37 +00:00
|
|
|
}()
|
|
|
|
}
|
2019-11-18 17:19:03 +00:00
|
|
|
|
|
|
|
go s.processAttestation()
|
2019-01-28 11:59:37 +00:00
|
|
|
}
|
|
|
|
|
2019-03-13 21:17:32 +00:00
|
|
|
// 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.
|
2019-11-23 03:35:47 +00:00
|
|
|
func (s *Service) processChainStartTime(ctx context.Context, genesisTime time.Time) {
|
2019-09-09 21:13:50 +00:00
|
|
|
initialDeposits := s.chainStartFetcher.ChainStartDeposits()
|
|
|
|
if err := s.initializeBeaconChain(ctx, genesisTime, initialDeposits, s.chainStartFetcher.ChainStartEth1Data()); err != nil {
|
2019-03-13 21:17:32 +00:00
|
|
|
log.Fatalf("Could not initialize beacon chain: %v", err)
|
|
|
|
}
|
2019-12-07 17:57:26 +00:00
|
|
|
s.stateNotifier.StateFeed().Send(&feed.Event{
|
|
|
|
Type: statefeed.Initialized,
|
|
|
|
Data: &statefeed.InitializedData{
|
2019-11-23 03:35:47 +00:00
|
|
|
StartTime: genesisTime,
|
|
|
|
},
|
|
|
|
})
|
2019-03-13 21:17:32 +00:00
|
|
|
}
|
|
|
|
|
2019-01-28 11:59:37 +00:00
|
|
|
// 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.
|
2019-09-07 02:39:14 +00:00
|
|
|
func (s *Service) initializeBeaconChain(
|
2019-08-24 01:59:09 +00:00
|
|
|
ctx context.Context,
|
|
|
|
genesisTime time.Time,
|
|
|
|
deposits []*ethpb.Deposit,
|
|
|
|
eth1data *ethpb.Eth1Data) error {
|
2019-09-07 02:39:14 +00:00
|
|
|
_, span := trace.StartSpan(context.Background(), "beacon-chain.Service.initializeBeaconChain")
|
2019-04-02 08:49:45 +00:00
|
|
|
defer span.End()
|
2019-10-01 20:05:17 +00:00
|
|
|
log.Info("Genesis time reached, starting the beacon chain")
|
2019-09-07 02:39:14 +00:00
|
|
|
s.genesisTime = genesisTime
|
2019-01-28 11:59:37 +00:00
|
|
|
unixTime := uint64(genesisTime.Unix())
|
2019-08-21 16:04:00 +00:00
|
|
|
|
2019-08-24 01:59:09 +00:00
|
|
|
genesisState, err := state.GenesisBeaconState(deposits, unixTime, eth1data)
|
2019-01-28 11:59:37 +00:00
|
|
|
if err != nil {
|
2019-08-22 01:14:24 +00:00
|
|
|
return errors.Wrap(err, "could not initialize genesis state")
|
2019-01-28 11:59:37 +00:00
|
|
|
}
|
2019-04-24 17:21:00 +00:00
|
|
|
|
2019-09-07 02:39:14 +00:00
|
|
|
if err := s.saveGenesisData(ctx, genesisState); err != nil {
|
2019-09-04 22:32:38 +00:00
|
|
|
return errors.Wrap(err, "could not save genesis data")
|
2019-08-29 22:32:35 +00:00
|
|
|
}
|
|
|
|
|
2019-09-27 22:56:08 +00:00
|
|
|
// Update committee shuffled indices for genesis epoch.
|
2019-10-07 05:11:49 +00:00
|
|
|
if featureconfig.Get().EnableNewCache {
|
2020-01-03 23:47:54 +00:00
|
|
|
if err := helpers.UpdateCommitteeCache(genesisState, 0 /* genesis epoch */); err != nil {
|
2019-09-27 22:56:08 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-22 01:14:24 +00:00
|
|
|
return nil
|
2018-07-19 16:31:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Stop the blockchain service's main event loop and associated goroutines.
|
2019-09-07 02:39:14 +00:00
|
|
|
func (s *Service) Stop() error {
|
|
|
|
defer s.cancel()
|
2018-07-19 16:31:50 +00:00
|
|
|
return nil
|
|
|
|
}
|
2018-07-22 16:58:14 +00:00
|
|
|
|
2019-09-16 20:45:03 +00:00
|
|
|
// Status always returns nil unless there is an error condition that causes
|
|
|
|
// this service to be unhealthy.
|
2019-09-07 02:39:14 +00:00
|
|
|
func (s *Service) Status() error {
|
|
|
|
if runtime.NumGoroutine() > int(s.maxRoutines) {
|
2019-05-15 14:38:27 +00:00
|
|
|
return fmt.Errorf("too many goroutines %d", runtime.NumGoroutine())
|
|
|
|
}
|
2018-12-30 21:20:43 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-08-23 01:13:56 +00:00
|
|
|
// This gets called to update canonical root mapping.
|
2019-09-07 02:39:14 +00:00
|
|
|
func (s *Service) saveHead(ctx context.Context, b *ethpb.BeaconBlock, r [32]byte) error {
|
2019-09-23 19:24:42 +00:00
|
|
|
s.headLock.Lock()
|
|
|
|
defer s.headLock.Unlock()
|
|
|
|
|
2019-09-07 02:39:14 +00:00
|
|
|
s.headSlot = b.Slot
|
2019-08-23 01:13:56 +00:00
|
|
|
|
2019-09-07 02:39:14 +00:00
|
|
|
s.canonicalRoots[b.Slot] = r[:]
|
2019-08-23 01:13:56 +00:00
|
|
|
|
2019-09-07 02:39:14 +00:00
|
|
|
if err := s.beaconDB.SaveHeadBlockRoot(ctx, r); err != nil {
|
2019-08-23 01:13:56 +00:00
|
|
|
return errors.Wrap(err, "could not save head root in DB")
|
|
|
|
}
|
2019-09-07 02:39:14 +00:00
|
|
|
s.headBlock = b
|
2019-08-23 01:13:56 +00:00
|
|
|
|
2019-09-07 02:39:14 +00:00
|
|
|
headState, err := s.beaconDB.State(ctx, r)
|
2019-08-23 01:13:56 +00:00
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "could not retrieve head state in DB")
|
|
|
|
}
|
2019-09-07 02:39:14 +00:00
|
|
|
s.headState = headState
|
2019-08-23 01:13:56 +00:00
|
|
|
|
|
|
|
log.WithFields(logrus.Fields{
|
2019-10-01 20:05:17 +00:00
|
|
|
"slot": b.Slot,
|
|
|
|
"headRoot": fmt.Sprintf("%#x", r),
|
2019-09-30 20:15:56 +00:00
|
|
|
}).Debug("Saved new head info")
|
2019-12-07 22:45:39 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// This gets called to update canonical root mapping. It does not save head block
|
|
|
|
// root in DB. With the inception of inital-sync-cache-state flag, it uses finalized
|
|
|
|
// check point as anchors to resume sync therefore head is no longer needed to be saved on per slot basis.
|
|
|
|
func (s *Service) saveHeadNoDB(ctx context.Context, b *ethpb.BeaconBlock, r [32]byte) error {
|
|
|
|
s.headLock.Lock()
|
|
|
|
defer s.headLock.Unlock()
|
|
|
|
|
|
|
|
s.headSlot = b.Slot
|
|
|
|
|
|
|
|
s.canonicalRoots[b.Slot] = r[:]
|
|
|
|
|
|
|
|
s.headBlock = b
|
|
|
|
|
|
|
|
headState, err := s.beaconDB.State(ctx, r)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "could not retrieve head state in DB")
|
|
|
|
}
|
|
|
|
s.headState = headState
|
|
|
|
|
|
|
|
log.WithFields(logrus.Fields{
|
|
|
|
"slot": b.Slot,
|
|
|
|
"headRoot": fmt.Sprintf("%#x", r),
|
|
|
|
}).Debug("Saved new head info")
|
2019-08-23 01:13:56 +00:00
|
|
|
return nil
|
|
|
|
}
|
2019-08-23 18:18:39 +00:00
|
|
|
|
2019-08-24 01:59:09 +00:00
|
|
|
// This gets called when beacon chain is first initialized to save validator indices and pubkeys in db
|
2019-09-07 02:39:14 +00:00
|
|
|
func (s *Service) saveGenesisValidators(ctx context.Context, state *pb.BeaconState) error {
|
|
|
|
for i, v := range state.Validators {
|
|
|
|
if err := s.beaconDB.SaveValidatorIndex(ctx, bytesutil.ToBytes48(v.PublicKey), uint64(i)); err != nil {
|
2019-08-24 01:59:09 +00:00
|
|
|
return errors.Wrapf(err, "could not save validator index: %d", i)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
2019-08-30 22:24:37 +00:00
|
|
|
|
2019-09-04 22:32:38 +00:00
|
|
|
// This gets called when beacon chain is first initialized to save genesis data (state, block, and more) in db
|
2019-09-07 02:39:14 +00:00
|
|
|
func (s *Service) saveGenesisData(ctx context.Context, genesisState *pb.BeaconState) error {
|
2019-09-23 19:24:42 +00:00
|
|
|
s.headLock.Lock()
|
|
|
|
defer s.headLock.Unlock()
|
|
|
|
|
2019-09-04 22:32:38 +00:00
|
|
|
stateRoot, err := ssz.HashTreeRoot(genesisState)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "could not tree hash genesis state")
|
|
|
|
}
|
|
|
|
genesisBlk := blocks.NewGenesisBlock(stateRoot[:])
|
|
|
|
genesisBlkRoot, err := ssz.SigningRoot(genesisBlk)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "could not get genesis block root")
|
|
|
|
}
|
|
|
|
|
2019-09-07 02:39:14 +00:00
|
|
|
if err := s.beaconDB.SaveBlock(ctx, genesisBlk); err != nil {
|
2019-09-04 22:32:38 +00:00
|
|
|
return errors.Wrap(err, "could not save genesis block")
|
|
|
|
}
|
2019-12-09 07:35:18 +00:00
|
|
|
if err := s.beaconDB.SaveState(ctx, genesisState, genesisBlkRoot); err != nil {
|
|
|
|
return errors.Wrap(err, "could not save genesis state")
|
|
|
|
}
|
2019-09-07 02:39:14 +00:00
|
|
|
if err := s.beaconDB.SaveHeadBlockRoot(ctx, genesisBlkRoot); err != nil {
|
2019-09-04 22:32:38 +00:00
|
|
|
return errors.Wrap(err, "could not save head block root")
|
|
|
|
}
|
2019-09-07 02:39:14 +00:00
|
|
|
if err := s.beaconDB.SaveGenesisBlockRoot(ctx, genesisBlkRoot); err != nil {
|
2019-09-04 22:32:38 +00:00
|
|
|
return errors.Wrap(err, "could save genesis block root")
|
|
|
|
}
|
2019-09-07 02:39:14 +00:00
|
|
|
if err := s.saveGenesisValidators(ctx, genesisState); err != nil {
|
2019-09-04 22:32:38 +00:00
|
|
|
return errors.Wrap(err, "could not save genesis validators")
|
|
|
|
}
|
|
|
|
|
|
|
|
genesisCheckpoint := ðpb.Checkpoint{Root: genesisBlkRoot[:]}
|
2019-09-07 02:39:14 +00:00
|
|
|
if err := s.forkChoiceStore.GenesisStore(ctx, genesisCheckpoint, genesisCheckpoint); err != nil {
|
2019-09-04 22:32:38 +00:00
|
|
|
return errors.Wrap(err, "Could not start fork choice service: %v")
|
|
|
|
}
|
|
|
|
|
2019-09-07 02:39:14 +00:00
|
|
|
s.headBlock = genesisBlk
|
|
|
|
s.headState = genesisState
|
|
|
|
s.canonicalRoots[genesisState.Slot] = genesisBlkRoot[:]
|
2019-09-04 22:32:38 +00:00
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-09-27 19:39:32 +00:00
|
|
|
// This gets called to initialize chain info variables using the finalized checkpoint stored in DB
|
2019-09-07 02:39:14 +00:00
|
|
|
func (s *Service) initializeChainInfo(ctx context.Context) error {
|
2019-09-23 19:24:42 +00:00
|
|
|
s.headLock.Lock()
|
|
|
|
defer s.headLock.Unlock()
|
|
|
|
|
2019-09-27 19:39:32 +00:00
|
|
|
finalized, err := s.beaconDB.FinalizedCheckpoint(ctx)
|
2019-08-30 22:24:37 +00:00
|
|
|
if err != nil {
|
2019-09-29 01:42:44 +00:00
|
|
|
return errors.Wrap(err, "could not get finalized checkpoint from db")
|
2019-08-30 22:24:37 +00:00
|
|
|
}
|
2019-09-29 01:42:44 +00:00
|
|
|
if finalized == nil {
|
2019-09-27 19:39:32 +00:00
|
|
|
// This should never happen. At chain start, the finalized checkpoint
|
|
|
|
// would be the genesis state and block.
|
|
|
|
return errors.New("no finalized epoch in the database")
|
|
|
|
}
|
|
|
|
s.headState, err = s.beaconDB.State(ctx, bytesutil.ToBytes32(finalized.Root))
|
2019-08-30 22:24:37 +00:00
|
|
|
if err != nil {
|
2019-09-27 19:39:32 +00:00
|
|
|
return errors.Wrap(err, "could not get finalized state from db")
|
2019-08-30 22:24:37 +00:00
|
|
|
}
|
2019-09-27 19:39:32 +00:00
|
|
|
s.headBlock, err = s.beaconDB.Block(ctx, bytesutil.ToBytes32(finalized.Root))
|
2019-08-30 22:24:37 +00:00
|
|
|
if err != nil {
|
2019-09-27 19:39:32 +00:00
|
|
|
return errors.Wrap(err, "could not get finalized block from db")
|
2019-08-30 22:24:37 +00:00
|
|
|
}
|
2019-09-27 19:39:32 +00:00
|
|
|
|
2019-12-06 00:49:19 +00:00
|
|
|
s.headSlot = s.headBlock.Slot
|
2019-09-27 19:39:32 +00:00
|
|
|
s.canonicalRoots[s.headSlot] = finalized.Root
|
2019-08-30 22:24:37 +00:00
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|