2018-08-09 18:25:48 +00:00
|
|
|
// Package blockchain defines the life-cycle and status of the beacon chain.
|
2018-07-19 16:31:50 +00:00
|
|
|
package blockchain
|
|
|
|
|
|
|
|
import (
|
2018-11-11 16:54:17 +00:00
|
|
|
"bytes"
|
2018-07-19 16:31:50 +00:00
|
|
|
"context"
|
2018-11-06 20:48:11 +00:00
|
|
|
"errors"
|
2018-07-31 04:41:27 +00:00
|
|
|
"fmt"
|
2018-09-18 13:06:28 +00:00
|
|
|
"time"
|
2018-07-19 16:31:50 +00:00
|
|
|
|
2018-07-22 16:58:14 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/powchain"
|
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/types"
|
2018-11-11 16:54:17 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/utils"
|
|
|
|
"github.com/prysmaticlabs/prysm/shared/bitutil"
|
2018-10-03 01:49:01 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/event"
|
2018-07-21 17:51:18 +00:00
|
|
|
"github.com/sirupsen/logrus"
|
2018-07-19 16:31:50 +00:00
|
|
|
)
|
|
|
|
|
2018-07-21 17:51:18 +00:00
|
|
|
var log = logrus.WithField("prefix", "blockchain")
|
|
|
|
|
2018-11-07 19:07:41 +00:00
|
|
|
type beaconDB interface {
|
|
|
|
GetBlock(h [32]byte) (*types.Block, error)
|
|
|
|
GetChainHead() (*types.Block, error)
|
|
|
|
GetActiveState() (*types.ActiveState, error)
|
|
|
|
GetCrystallizedState() (*types.CrystallizedState, error)
|
|
|
|
GetGenesisTime() (time.Time, error)
|
|
|
|
HasBlock(h [32]byte) bool
|
2018-11-11 16:54:17 +00:00
|
|
|
ReadBlockVoteCache(blockHashes [][32]byte) (utils.BlockVoteCache, error)
|
2018-11-07 19:07:41 +00:00
|
|
|
SaveBlock(block *types.Block) error
|
|
|
|
SaveUnfinalizedBlockState(aState *types.ActiveState, cState *types.CrystallizedState) error
|
|
|
|
UpdateChainHead(head *types.Block, aState *types.ActiveState, cState *types.CrystallizedState) error
|
2018-11-11 16:54:17 +00:00
|
|
|
WriteBlockVoteCache(blockVoteCache utils.BlockVoteCache) error
|
2018-11-07 19:07:41 +00:00
|
|
|
}
|
|
|
|
|
2018-07-19 16:31:50 +00:00
|
|
|
// ChainService represents a service that handles the internal
|
|
|
|
// logic of managing the full PoS beacon chain.
|
|
|
|
type ChainService struct {
|
2018-08-24 04:09:59 +00:00
|
|
|
ctx context.Context
|
|
|
|
cancel context.CancelFunc
|
2018-11-07 19:07:41 +00:00
|
|
|
beaconDB beaconDB
|
2018-08-24 04:09:59 +00:00
|
|
|
web3Service *powchain.Web3Service
|
|
|
|
incomingBlockFeed *event.Feed
|
|
|
|
incomingBlockChan chan *types.Block
|
2018-10-14 15:29:57 +00:00
|
|
|
processedBlockChan chan *types.Block
|
2018-08-24 04:09:59 +00:00
|
|
|
canonicalBlockFeed *event.Feed
|
|
|
|
canonicalCrystallizedStateFeed *event.Feed
|
2018-10-10 16:17:48 +00:00
|
|
|
genesisTime time.Time
|
2018-10-18 04:23:18 +00:00
|
|
|
unfinalizedBlocks map[[32]byte]*statePair
|
2018-10-02 02:04:37 +00:00
|
|
|
enablePOWChain bool
|
2018-07-19 16:31:50 +00:00
|
|
|
}
|
|
|
|
|
2018-08-09 22:54:59 +00:00
|
|
|
// Config options for the service.
|
|
|
|
type Config struct {
|
2018-11-05 17:35:50 +00:00
|
|
|
BeaconBlockBuf int
|
|
|
|
IncomingBlockBuf int
|
|
|
|
Web3Service *powchain.Web3Service
|
2018-11-07 19:07:41 +00:00
|
|
|
BeaconDB beaconDB
|
2018-11-05 17:35:50 +00:00
|
|
|
DevMode bool
|
|
|
|
EnablePOWChain bool
|
2018-08-09 22:54:59 +00:00
|
|
|
}
|
|
|
|
|
2018-10-18 04:23:18 +00:00
|
|
|
// Struct used to represent an unfinalized block's state pair
|
|
|
|
// (active state, crystallized state) tuple.
|
|
|
|
type statePair struct {
|
|
|
|
crystallizedState *types.CrystallizedState
|
|
|
|
activeState *types.ActiveState
|
|
|
|
cycleTransition bool
|
|
|
|
}
|
|
|
|
|
2018-07-19 16:31:50 +00:00
|
|
|
// NewChainService instantiates a new service instance that will
|
|
|
|
// be registered into a running beacon node.
|
2018-08-24 04:09:59 +00:00
|
|
|
func NewChainService(ctx context.Context, cfg *Config) (*ChainService, error) {
|
2018-07-19 16:31:50 +00:00
|
|
|
ctx, cancel := context.WithCancel(ctx)
|
2018-07-31 04:41:27 +00:00
|
|
|
return &ChainService{
|
2018-09-02 16:44:03 +00:00
|
|
|
ctx: ctx,
|
|
|
|
cancel: cancel,
|
|
|
|
beaconDB: cfg.BeaconDB,
|
|
|
|
web3Service: cfg.Web3Service,
|
|
|
|
incomingBlockChan: make(chan *types.Block, cfg.IncomingBlockBuf),
|
2018-10-14 15:29:57 +00:00
|
|
|
processedBlockChan: make(chan *types.Block),
|
2018-09-02 16:44:03 +00:00
|
|
|
incomingBlockFeed: new(event.Feed),
|
|
|
|
canonicalBlockFeed: new(event.Feed),
|
|
|
|
canonicalCrystallizedStateFeed: new(event.Feed),
|
2018-10-18 04:23:18 +00:00
|
|
|
unfinalizedBlocks: make(map[[32]byte]*statePair),
|
2018-10-02 02:04:37 +00:00
|
|
|
enablePOWChain: cfg.EnablePOWChain,
|
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.
|
|
|
|
func (c *ChainService) Start() {
|
2018-09-27 02:34:35 +00:00
|
|
|
log.Info("Starting service")
|
|
|
|
|
2018-10-10 16:17:48 +00:00
|
|
|
var err error
|
|
|
|
c.genesisTime, err = c.beaconDB.GetGenesisTime()
|
2018-10-02 20:07:33 +00:00
|
|
|
if err != nil {
|
2018-10-10 16:17:48 +00:00
|
|
|
log.Fatal(err)
|
|
|
|
return
|
2018-10-02 20:07:33 +00:00
|
|
|
}
|
|
|
|
|
2018-10-18 04:23:18 +00:00
|
|
|
// TODO(#675): Initialize unfinalizedBlocks map from disk in case this
|
|
|
|
// is a beacon node restarting.
|
2018-10-14 15:29:57 +00:00
|
|
|
go c.updateHead(c.processedBlockChan)
|
|
|
|
go c.blockProcessing(c.processedBlockChan)
|
2018-07-19 16:31:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Stop the blockchain service's main event loop and associated goroutines.
|
|
|
|
func (c *ChainService) Stop() error {
|
|
|
|
defer c.cancel()
|
2018-10-10 16:17:48 +00:00
|
|
|
|
2018-07-25 16:57:44 +00:00
|
|
|
log.Info("Stopping service")
|
2018-07-19 16:31:50 +00:00
|
|
|
return nil
|
|
|
|
}
|
2018-07-22 16:58:14 +00:00
|
|
|
|
2018-09-11 17:08:31 +00:00
|
|
|
// IncomingBlockFeed returns a feed that any service can send incoming p2p blocks into.
|
2018-08-24 04:09:59 +00:00
|
|
|
// The chain service will subscribe to this feed in order to process incoming blocks.
|
|
|
|
func (c *ChainService) IncomingBlockFeed() *event.Feed {
|
|
|
|
return c.incomingBlockFeed
|
|
|
|
}
|
|
|
|
|
|
|
|
// CanonicalBlockFeed returns a channel that is written to
|
2018-08-18 03:34:56 +00:00
|
|
|
// whenever a new block is determined to be canonical in the chain.
|
2018-08-24 04:09:59 +00:00
|
|
|
func (c *ChainService) CanonicalBlockFeed() *event.Feed {
|
|
|
|
return c.canonicalBlockFeed
|
2018-08-18 03:34:56 +00:00
|
|
|
}
|
|
|
|
|
2018-08-24 04:09:59 +00:00
|
|
|
// CanonicalCrystallizedStateFeed returns a feed that is written to
|
2018-08-18 03:34:56 +00:00
|
|
|
// whenever a new crystallized state is determined to be canonical in the chain.
|
2018-08-24 04:09:59 +00:00
|
|
|
func (c *ChainService) CanonicalCrystallizedStateFeed() *event.Feed {
|
|
|
|
return c.canonicalCrystallizedStateFeed
|
|
|
|
}
|
|
|
|
|
2018-09-11 05:09:41 +00:00
|
|
|
// doesPoWBlockExist checks if the referenced PoW block exists.
|
|
|
|
func (c *ChainService) doesPoWBlockExist(block *types.Block) bool {
|
|
|
|
powBlock, err := c.web3Service.Client().BlockByHash(context.Background(), block.PowChainRef())
|
|
|
|
if err != nil {
|
|
|
|
log.Debugf("fetching PoW block corresponding to mainchain reference failed: %v", err)
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
return powBlock != nil
|
|
|
|
}
|
|
|
|
|
2018-09-18 13:06:28 +00:00
|
|
|
// updateHead applies the fork choice rule to the beacon chain
|
|
|
|
// at the start of each new slot interval. The function looks
|
|
|
|
// at an in-memory slice of block hashes pending processing and
|
|
|
|
// selects the best block according to the in-protocol fork choice
|
|
|
|
// rule as canonical. This block is then persisted to storage.
|
2018-10-14 15:29:57 +00:00
|
|
|
func (c *ChainService) updateHead(processedBlock <-chan *types.Block) {
|
2018-09-18 13:06:28 +00:00
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case <-c.ctx.Done():
|
|
|
|
return
|
2018-10-14 15:29:57 +00:00
|
|
|
case block := <-processedBlock:
|
2018-11-06 20:48:11 +00:00
|
|
|
if block == nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2018-09-18 13:06:28 +00:00
|
|
|
h, err := block.Hash()
|
|
|
|
if err != nil {
|
|
|
|
log.Errorf("Could not hash incoming block: %v", err)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2018-10-18 04:23:18 +00:00
|
|
|
log.Info("Updating chain head...")
|
|
|
|
currentHead, err := c.beaconDB.GetChainHead()
|
2018-09-21 19:33:53 +00:00
|
|
|
if err != nil {
|
2018-10-18 04:23:18 +00:00
|
|
|
log.Errorf("Could not get current chain head: %v", err)
|
2018-09-21 19:33:53 +00:00
|
|
|
continue
|
|
|
|
}
|
2018-10-18 04:23:18 +00:00
|
|
|
currentcState, err := c.beaconDB.GetCrystallizedState()
|
2018-10-17 06:11:24 +00:00
|
|
|
if err != nil {
|
2018-10-18 04:23:18 +00:00
|
|
|
log.Errorf("Could not get current crystallized state: %v", err)
|
|
|
|
continue
|
2018-10-17 06:11:24 +00:00
|
|
|
}
|
2018-11-05 17:35:50 +00:00
|
|
|
|
2018-10-18 04:23:18 +00:00
|
|
|
blockcState := c.unfinalizedBlocks[h].crystallizedState
|
|
|
|
|
|
|
|
var headUpdated bool
|
|
|
|
newHead := currentHead
|
|
|
|
// If both blocks have the same crystallized state root, we favor one which has
|
|
|
|
// the higher slot.
|
|
|
|
if currentHead.CrystallizedStateRoot() == block.CrystallizedStateRoot() {
|
|
|
|
if block.SlotNumber() > currentHead.SlotNumber() {
|
|
|
|
newHead = block
|
|
|
|
headUpdated = true
|
|
|
|
}
|
|
|
|
// 2a. Pick the block with the higher last_finalized_slot.
|
|
|
|
// 2b. If same, pick the block with the higher last_justified_slot.
|
2018-10-23 16:07:43 +00:00
|
|
|
} else if blockcState.LastFinalizedSlot() > currentcState.LastFinalizedSlot() {
|
|
|
|
newHead = block
|
|
|
|
headUpdated = true
|
|
|
|
} else if blockcState.LastFinalizedSlot() == currentcState.LastFinalizedSlot() {
|
|
|
|
if blockcState.LastJustifiedSlot() > currentcState.LastJustifiedSlot() {
|
2018-10-18 04:23:18 +00:00
|
|
|
newHead = block
|
|
|
|
headUpdated = true
|
2018-10-23 16:07:43 +00:00
|
|
|
} else if blockcState.LastJustifiedSlot() == currentcState.LastJustifiedSlot() {
|
|
|
|
if block.SlotNumber() > currentHead.SlotNumber() {
|
2018-10-18 04:23:18 +00:00
|
|
|
newHead = block
|
|
|
|
headUpdated = true
|
|
|
|
}
|
2018-09-18 13:06:28 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-18 04:23:18 +00:00
|
|
|
// If no new head was found, we do not update the chain.
|
|
|
|
if !headUpdated {
|
|
|
|
log.Info("Chain head not updated")
|
2018-09-18 13:06:28 +00:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2018-10-18 04:23:18 +00:00
|
|
|
// TODO(#674): Handle chain reorgs.
|
2018-10-17 06:11:24 +00:00
|
|
|
var newCState *types.CrystallizedState
|
2018-10-18 04:23:18 +00:00
|
|
|
if c.unfinalizedBlocks[h].cycleTransition {
|
|
|
|
newCState = blockcState
|
2018-09-18 13:06:28 +00:00
|
|
|
}
|
2018-10-18 04:23:18 +00:00
|
|
|
if err := c.beaconDB.UpdateChainHead(newHead, c.unfinalizedBlocks[h].activeState, newCState); err != nil {
|
2018-10-17 06:11:24 +00:00
|
|
|
log.Errorf("Failed to update chain: %v", err)
|
2018-10-18 04:23:18 +00:00
|
|
|
continue
|
2018-09-18 13:06:28 +00:00
|
|
|
}
|
2018-10-18 04:23:18 +00:00
|
|
|
log.WithField("blockHash", fmt.Sprintf("0x%x", h)).Info("Chain head block and state updated")
|
2018-09-18 13:06:28 +00:00
|
|
|
// We fire events that notify listeners of a new block (or crystallized state in
|
|
|
|
// the case of a state transition). This is useful for the beacon node's gRPC
|
|
|
|
// server to stream these events to beacon clients.
|
2018-10-18 04:23:18 +00:00
|
|
|
if c.unfinalizedBlocks[h].cycleTransition {
|
|
|
|
c.canonicalCrystallizedStateFeed.Send(blockcState)
|
2018-09-18 13:06:28 +00:00
|
|
|
}
|
2018-10-18 04:23:18 +00:00
|
|
|
c.canonicalBlockFeed.Send(newHead)
|
2018-09-18 13:06:28 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-03 12:50:14 +00:00
|
|
|
func (c *ChainService) executeStateTransition(
|
|
|
|
cState *types.CrystallizedState,
|
|
|
|
aState *types.ActiveState,
|
|
|
|
block *types.Block) (*types.CrystallizedState, error) {
|
|
|
|
|
|
|
|
var err error
|
|
|
|
log.Infof("Executing state transition for slot: %d", block.SlotNumber())
|
|
|
|
for cState.IsCycleTransition(block.SlotNumber()) {
|
2018-11-11 16:54:17 +00:00
|
|
|
cState, err = cState.NewStateRecalculations(aState, block, c.beaconDB)
|
2018-11-03 12:50:14 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return cState, nil
|
|
|
|
}
|
|
|
|
|
2018-10-14 15:29:57 +00:00
|
|
|
func (c *ChainService) blockProcessing(processedBlock chan<- *types.Block) {
|
2018-09-24 01:22:09 +00:00
|
|
|
subBlock := c.incomingBlockFeed.Subscribe(c.incomingBlockChan)
|
|
|
|
defer subBlock.Unsubscribe()
|
2018-07-22 16:58:14 +00:00
|
|
|
for {
|
|
|
|
select {
|
2018-09-18 13:06:28 +00:00
|
|
|
case <-c.ctx.Done():
|
2018-08-24 04:09:59 +00:00
|
|
|
log.Debug("Chain service context closed, exiting goroutine")
|
|
|
|
return
|
2018-09-11 17:08:31 +00:00
|
|
|
|
2018-08-24 04:09:59 +00:00
|
|
|
// Listen for a newly received incoming block from the sync service.
|
|
|
|
case block := <-c.incomingBlockChan:
|
2018-11-06 20:48:11 +00:00
|
|
|
if err := c.processBlock(block); err != nil {
|
|
|
|
log.Error(err)
|
|
|
|
processedBlock <- nil
|
2018-08-24 04:09:59 +00:00
|
|
|
continue
|
2018-08-15 04:49:59 +00:00
|
|
|
}
|
2018-08-18 03:34:56 +00:00
|
|
|
|
2018-11-06 20:48:11 +00:00
|
|
|
// Push the block to trigger the fork choice rule.
|
|
|
|
processedBlock <- block
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-09-15 14:51:17 +00:00
|
|
|
|
2018-11-06 20:48:11 +00:00
|
|
|
func (c *ChainService) processBlock(block *types.Block) error {
|
|
|
|
blockHash, err := block.Hash()
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("Failed to get hash of block: %v", err)
|
|
|
|
}
|
2018-10-17 06:11:24 +00:00
|
|
|
|
2018-11-06 20:48:11 +00:00
|
|
|
if c.enablePOWChain && !c.doesPoWBlockExist(block) {
|
|
|
|
return errors.New("Proof-of-Work chain reference in block does not exist")
|
|
|
|
}
|
2018-09-21 19:33:53 +00:00
|
|
|
|
2018-11-06 20:48:11 +00:00
|
|
|
parent, err := c.beaconDB.GetBlock(block.ParentHash())
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("Could not get parent block: %v", err)
|
|
|
|
}
|
|
|
|
if parent == nil {
|
|
|
|
return fmt.Errorf("Block points to nil parent: %#x", block.ParentHash())
|
|
|
|
}
|
2018-09-04 23:18:55 +00:00
|
|
|
|
2018-11-06 20:48:11 +00:00
|
|
|
aState, err := c.beaconDB.GetActiveState()
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("Failed to get active state: %v", err)
|
|
|
|
}
|
|
|
|
cState, err := c.beaconDB.GetCrystallizedState()
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("Failed to get crystallized state: %v", err)
|
|
|
|
}
|
2018-10-18 04:23:18 +00:00
|
|
|
|
2018-11-06 20:48:11 +00:00
|
|
|
if valid := block.IsValid(
|
|
|
|
c.beaconDB,
|
|
|
|
aState,
|
|
|
|
cState,
|
|
|
|
parent.SlotNumber(),
|
|
|
|
c.genesisTime,
|
|
|
|
); !valid {
|
|
|
|
return errors.New("Block failed validity conditions")
|
|
|
|
}
|
2018-10-18 04:23:18 +00:00
|
|
|
|
2018-11-11 16:54:17 +00:00
|
|
|
if err = c.calculateNewBlockVotes(block, aState, cState); err != nil {
|
|
|
|
return fmt.Errorf("Failed to update block vote cache: %v", err)
|
|
|
|
}
|
|
|
|
|
2018-11-08 18:13:50 +00:00
|
|
|
// First, include new attestations to the active state
|
|
|
|
// so that they're accounted for during cycle transitions.
|
|
|
|
aState = aState.UpdateAttestations(block.Attestations())
|
|
|
|
|
2018-11-06 20:48:11 +00:00
|
|
|
// If the block is valid, we compute its associated state tuple (active, crystallized)
|
|
|
|
// and apply a block scoring function.
|
|
|
|
var didCycleTransition bool
|
|
|
|
if cState.IsCycleTransition(block.SlotNumber()) {
|
|
|
|
cState, err = c.executeStateTransition(cState, aState, block)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("Initialize new cycle transition failed: %v", err)
|
|
|
|
}
|
|
|
|
didCycleTransition = true
|
|
|
|
}
|
2018-09-02 16:44:03 +00:00
|
|
|
|
2018-11-06 20:48:11 +00:00
|
|
|
aState, err = aState.CalculateNewActiveState(
|
|
|
|
block,
|
|
|
|
cState,
|
|
|
|
parent.SlotNumber(),
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("Compute active state failed: %v", err)
|
|
|
|
}
|
2018-09-02 16:44:03 +00:00
|
|
|
|
2018-11-06 20:48:11 +00:00
|
|
|
if err := c.beaconDB.SaveBlock(block); err != nil {
|
|
|
|
return fmt.Errorf("Failed to save block: %v", err)
|
|
|
|
}
|
|
|
|
if err := c.beaconDB.SaveUnfinalizedBlockState(aState, cState); err != nil {
|
|
|
|
return fmt.Errorf("Error persisting unfinalized block's state: %v", err)
|
|
|
|
}
|
2018-10-18 04:23:18 +00:00
|
|
|
|
2018-11-06 20:48:11 +00:00
|
|
|
log.Infof("Processed block: %#x", blockHash)
|
|
|
|
|
|
|
|
c.unfinalizedBlocks[blockHash] = &statePair{
|
|
|
|
crystallizedState: cState,
|
|
|
|
activeState: aState,
|
|
|
|
cycleTransition: didCycleTransition,
|
2018-07-22 16:58:14 +00:00
|
|
|
}
|
2018-11-06 20:48:11 +00:00
|
|
|
|
|
|
|
return nil
|
2018-07-22 16:58:14 +00:00
|
|
|
}
|
2018-11-11 16:54:17 +00:00
|
|
|
|
|
|
|
func (c *ChainService) calculateNewBlockVotes(block *types.Block, aState *types.ActiveState, cState *types.CrystallizedState) error {
|
|
|
|
for _, attestation := range block.Attestations() {
|
|
|
|
parentHashes, err := aState.GetSignedParentHashes(block, attestation)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
attesterIndices, err := cState.AttesterIndices(attestation)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Read block vote cache from DB.
|
|
|
|
var blockVoteCache utils.BlockVoteCache
|
|
|
|
if blockVoteCache, err = c.beaconDB.ReadBlockVoteCache(parentHashes); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update block vote cache.
|
|
|
|
for _, h := range parentHashes {
|
|
|
|
// Skip calculating for this hash if the hash is part of oblique parent hashes.
|
|
|
|
var skip bool
|
|
|
|
for _, oblique := range attestation.ObliqueParentHashes {
|
|
|
|
if bytes.Equal(h[:], oblique) {
|
|
|
|
skip = true
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if skip {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
// Initialize vote cache of a given block hash if it doesn't exist already.
|
|
|
|
if !blockVoteCache.IsVoteCacheExist(h) {
|
|
|
|
blockVoteCache[h] = utils.NewBlockVote()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Loop through attester indices, if the attester has voted but was not accounted for
|
|
|
|
// in the cache, then we add attester's index and balance to the block cache.
|
|
|
|
for i, attesterIndex := range attesterIndices {
|
|
|
|
var attesterExists bool
|
|
|
|
isBitSet, err := bitutil.CheckBit(attestation.AttesterBitfield, i)
|
|
|
|
if err != nil {
|
|
|
|
log.Errorf("Bitfield check for cache adding failed at index: %d with: %v", i, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if !isBitSet {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
for _, indexInCache := range blockVoteCache[h].VoterIndices {
|
|
|
|
if attesterIndex == indexInCache {
|
|
|
|
attesterExists = true
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if !attesterExists {
|
|
|
|
blockVoteCache[h].VoterIndices = append(blockVoteCache[h].VoterIndices, attesterIndex)
|
|
|
|
blockVoteCache[h].VoteTotalDeposit += cState.Validators()[attesterIndex].Balance
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-14 20:24:34 +00:00
|
|
|
// Write updated block vote cache back to DB.
|
2018-11-11 16:54:17 +00:00
|
|
|
if err = c.beaconDB.WriteBlockVoteCache(blockVoteCache); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|