2021-09-10 03:52:56 +00:00
|
|
|
package sync
|
|
|
|
|
|
|
|
import (
|
|
|
|
"github.com/pkg/errors"
|
2023-03-17 18:52:56 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/p2p"
|
|
|
|
"github.com/prysmaticlabs/prysm/v4/config/params"
|
|
|
|
"github.com/prysmaticlabs/prysm/v4/consensus-types/primitives"
|
|
|
|
"github.com/prysmaticlabs/prysm/v4/network/forks"
|
|
|
|
"github.com/prysmaticlabs/prysm/v4/time/slots"
|
2021-09-10 03:52:56 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// Is a background routine that observes for new incoming forks. Depending on the epoch
|
|
|
|
// it will be in charge of subscribing/unsubscribing the relevant topics at the fork boundaries.
|
|
|
|
func (s *Service) forkWatcher() {
|
2023-05-03 04:34:01 +00:00
|
|
|
slotTicker := slots.NewSlotTicker(s.cfg.clock.GenesisTime(), params.BeaconConfig().SecondsPerSlot)
|
2021-09-10 03:52:56 +00:00
|
|
|
for {
|
|
|
|
select {
|
|
|
|
// In the event of a node restart, we will still end up subscribing to the correct
|
|
|
|
// topics during/after the fork epoch. This routine is to ensure correct
|
|
|
|
// subscriptions for nodes running before a fork epoch.
|
|
|
|
case currSlot := <-slotTicker.C():
|
2021-10-01 20:17:57 +00:00
|
|
|
currEpoch := slots.ToEpoch(currSlot)
|
2021-09-10 03:52:56 +00:00
|
|
|
if err := s.registerForUpcomingFork(currEpoch); err != nil {
|
|
|
|
log.WithError(err).Error("Unable to check for fork in the next epoch")
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if err := s.deregisterFromPastFork(currEpoch); err != nil {
|
|
|
|
log.WithError(err).Error("Unable to check for fork in the previous epoch")
|
|
|
|
continue
|
|
|
|
}
|
2023-01-16 15:00:11 +00:00
|
|
|
// Broadcast BLS changes at the Capella fork boundary
|
2023-01-26 23:13:28 +00:00
|
|
|
s.broadcastBLSChanges(currSlot)
|
2023-01-16 15:00:11 +00:00
|
|
|
|
2021-09-10 03:52:56 +00:00
|
|
|
case <-s.ctx.Done():
|
|
|
|
log.Debug("Context closed, exiting goroutine")
|
|
|
|
slotTicker.Done()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Checks if there is a fork in the next epoch and if there is
|
|
|
|
// it registers the appropriate gossip and rpc topics.
|
2023-01-26 14:40:12 +00:00
|
|
|
func (s *Service) registerForUpcomingFork(currEpoch primitives.Epoch) error {
|
2023-05-03 04:34:01 +00:00
|
|
|
genRoot := s.cfg.clock.GenesisValidatorsRoot()
|
|
|
|
isNextForkEpoch, err := forks.IsForkNextEpoch(s.cfg.clock.GenesisTime(), genRoot[:])
|
2021-09-10 03:52:56 +00:00
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "Could not retrieve next fork epoch")
|
|
|
|
}
|
|
|
|
// In preparation for the upcoming fork
|
|
|
|
// in the following epoch, the node
|
|
|
|
// will subscribe the new topics in advance.
|
|
|
|
if isNextForkEpoch {
|
|
|
|
nextEpoch := currEpoch + 1
|
2022-11-26 19:07:05 +00:00
|
|
|
digest, err := forks.ForkDigestFromEpoch(nextEpoch, genRoot[:])
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "could not retrieve fork digest")
|
|
|
|
}
|
|
|
|
if s.subHandler.digestExists(digest) {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
s.registerSubscribers(nextEpoch, digest)
|
|
|
|
if nextEpoch == params.BeaconConfig().AltairForkEpoch {
|
2021-09-10 03:52:56 +00:00
|
|
|
s.registerRPCHandlersAltair()
|
|
|
|
}
|
2023-09-18 14:30:34 +00:00
|
|
|
if nextEpoch == params.BeaconConfig().DenebForkEpoch {
|
|
|
|
s.registerRPCHandlersDeneb()
|
|
|
|
}
|
2021-09-10 03:52:56 +00:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Checks if there was a fork in the previous epoch, and if there
|
|
|
|
// was then we deregister the topics from that particular fork.
|
2023-01-26 14:40:12 +00:00
|
|
|
func (s *Service) deregisterFromPastFork(currEpoch primitives.Epoch) error {
|
2023-05-03 04:34:01 +00:00
|
|
|
genRoot := s.cfg.clock.GenesisValidatorsRoot()
|
2021-09-10 03:52:56 +00:00
|
|
|
// This method takes care of the de-registration of
|
|
|
|
// old gossip pubsub handlers. Once we are at the epoch
|
|
|
|
// after the fork, we de-register from all the outdated topics.
|
2021-09-17 19:20:50 +00:00
|
|
|
currFork, err := forks.Fork(currEpoch)
|
2021-09-10 03:52:56 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
// If we are still in our genesis fork version then
|
|
|
|
// we simply exit early.
|
|
|
|
if currFork.Epoch == params.BeaconConfig().GenesisEpoch {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
epochAfterFork := currFork.Epoch + 1
|
|
|
|
// If we are in the epoch after the fork, we start de-registering.
|
|
|
|
if epochAfterFork == currEpoch {
|
|
|
|
// Look at the previous fork's digest.
|
|
|
|
epochBeforeFork := currFork.Epoch - 1
|
2021-09-17 19:20:50 +00:00
|
|
|
prevDigest, err := forks.ForkDigestFromEpoch(epochBeforeFork, genRoot[:])
|
2021-09-10 03:52:56 +00:00
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "Failed to determine previous epoch fork digest")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Exit early if there are no topics with that particular
|
|
|
|
// digest.
|
|
|
|
if !s.subHandler.digestExists(prevDigest) {
|
|
|
|
return nil
|
|
|
|
}
|
2022-02-10 16:15:55 +00:00
|
|
|
prevFork, err := forks.Fork(epochBeforeFork)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "failed to determine previous epoch fork data")
|
|
|
|
}
|
2022-11-26 19:07:05 +00:00
|
|
|
if prevFork.Epoch == params.BeaconConfig().GenesisEpoch {
|
2022-02-10 16:15:55 +00:00
|
|
|
s.unregisterPhase0Handlers()
|
|
|
|
}
|
2021-09-10 03:52:56 +00:00
|
|
|
// Run through all our current active topics and see
|
|
|
|
// if there are any subscriptions to be removed.
|
|
|
|
for _, t := range s.subHandler.allTopics() {
|
|
|
|
retDigest, err := p2p.ExtractGossipDigest(t)
|
|
|
|
if err != nil {
|
|
|
|
log.WithError(err).Error("Could not retrieve digest")
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if retDigest == prevDigest {
|
|
|
|
s.unSubscribeFromTopic(t)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|