mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2024-12-31 23:41:22 +00:00
07e7e030d9
* Update pubsub and fix topicIDs * WIP filter * Add suggested code from @bidlocode * add tests and fix bugs * more tests * Wait until state initialized to accept pubsub filtering * rename for clarity and clarify comment * fix test builds * Autofix issues in 2 files Resolved issues in the following files via DeepSource Autofix: 1. beacon-chain/p2p/pubsub_filter.go 2. beacon-chain/p2p/pubsub_filter_test.go * @nisdas pr feedback * pr feedback and fuzz fix * Update beacon-chain/p2p/pubsub_filter.go * Must have protocol suffix * Must have protocol suffix * gofmt * rm test, fix panic * Fix tests * Add isInitialized check * Add a few more tests for better coverage * cache fork digest, make pubsub filter part of the p2p service * rename service * gofmt * Add comment * fix Co-authored-by: deepsource-autofix[bot] <62050782+deepsource-autofix[bot]@users.noreply.github.com> Co-authored-by: Nishant Das <nishdas93@gmail.com> Co-authored-by: prylabs-bulldozer[bot] <58059840+prylabs-bulldozer[bot]@users.noreply.github.com>
140 lines
4.4 KiB
Go
140 lines
4.4 KiB
Go
package p2p
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"math"
|
|
"time"
|
|
|
|
"github.com/ethereum/go-ethereum/p2p/enode"
|
|
"github.com/ethereum/go-ethereum/p2p/enr"
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/core/helpers"
|
|
pb "github.com/prysmaticlabs/prysm/proto/beacon/p2p/v1"
|
|
"github.com/prysmaticlabs/prysm/shared/p2putils"
|
|
"github.com/prysmaticlabs/prysm/shared/params"
|
|
"github.com/prysmaticlabs/prysm/shared/timeutils"
|
|
"github.com/sirupsen/logrus"
|
|
)
|
|
|
|
// ENR key used for eth2-related fork data.
|
|
var eth2ENRKey = params.BeaconNetworkConfig().ETH2Key
|
|
|
|
// ForkDigest returns the current fork digest of
|
|
// the node.
|
|
func (s *Service) forkDigest() ([4]byte, error) {
|
|
if s.currentForkDigest != [4]byte{} {
|
|
return s.currentForkDigest, nil
|
|
}
|
|
fd, err := p2putils.CreateForkDigest(s.genesisTime, s.genesisValidatorsRoot)
|
|
if err != nil {
|
|
s.currentForkDigest = fd
|
|
}
|
|
return fd, err
|
|
}
|
|
|
|
// Compares fork ENRs between an incoming peer's record and our node's
|
|
// local record values for current and next fork version/epoch.
|
|
func (s *Service) compareForkENR(record *enr.Record) error {
|
|
currentRecord := s.dv5Listener.LocalNode().Node().Record()
|
|
peerForkENR, err := retrieveForkEntry(record)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
currentForkENR, err := retrieveForkEntry(currentRecord)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
enrString, err := SerializeENR(record)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
// Clients SHOULD connect to peers with current_fork_digest, next_fork_version,
|
|
// and next_fork_epoch that match local values.
|
|
if !bytes.Equal(peerForkENR.CurrentForkDigest, currentForkENR.CurrentForkDigest) {
|
|
return fmt.Errorf(
|
|
"fork digest of peer with ENR %s: %v, does not match local value: %v",
|
|
enrString,
|
|
peerForkENR.CurrentForkDigest,
|
|
currentForkENR.CurrentForkDigest,
|
|
)
|
|
}
|
|
// Clients MAY connect to peers with the same current_fork_version but a
|
|
// different next_fork_version/next_fork_epoch. Unless ENRForkID is manually
|
|
// updated to matching prior to the earlier next_fork_epoch of the two clients,
|
|
// these type of connecting clients will be unable to successfully interact
|
|
// starting at the earlier next_fork_epoch.
|
|
if peerForkENR.NextForkEpoch != currentForkENR.NextForkEpoch {
|
|
log.WithFields(logrus.Fields{
|
|
"peerNextForkEpoch": peerForkENR.NextForkEpoch,
|
|
"peerENR": enrString,
|
|
}).Debug("Peer matches fork digest but has different next fork epoch")
|
|
}
|
|
if !bytes.Equal(peerForkENR.NextForkVersion, currentForkENR.NextForkVersion) {
|
|
log.WithFields(logrus.Fields{
|
|
"peerNextForkVersion": peerForkENR.NextForkVersion,
|
|
"peerENR": enrString,
|
|
}).Debug("Peer matches fork digest but has different next fork version")
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// Adds a fork entry as an ENR record under the eth2EnrKey for
|
|
// the local node. The fork entry is an ssz-encoded enrForkID type
|
|
// which takes into account the current fork version from the current
|
|
// epoch to create a fork digest, the next fork version,
|
|
// and the next fork epoch.
|
|
func addForkEntry(
|
|
node *enode.LocalNode,
|
|
genesisTime time.Time,
|
|
genesisValidatorsRoot []byte,
|
|
) (*enode.LocalNode, error) {
|
|
digest, err := p2putils.CreateForkDigest(genesisTime, genesisValidatorsRoot)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
currentSlot := helpers.SlotsSince(genesisTime)
|
|
currentEpoch := helpers.SlotToEpoch(currentSlot)
|
|
if timeutils.Now().Before(genesisTime) {
|
|
currentEpoch = 0
|
|
}
|
|
fork, err := p2putils.Fork(currentEpoch)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
nextForkEpoch := params.BeaconConfig().NextForkEpoch
|
|
nextForkVersion := params.BeaconConfig().NextForkVersion
|
|
// Set to the current fork version if our next fork is not planned.
|
|
if nextForkEpoch == math.MaxUint64 {
|
|
nextForkVersion = fork.CurrentVersion
|
|
}
|
|
enrForkID := &pb.ENRForkID{
|
|
CurrentForkDigest: digest[:],
|
|
NextForkVersion: nextForkVersion,
|
|
NextForkEpoch: nextForkEpoch,
|
|
}
|
|
enc, err := enrForkID.MarshalSSZ()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
forkEntry := enr.WithEntry(eth2ENRKey, enc)
|
|
node.Set(forkEntry)
|
|
return node, nil
|
|
}
|
|
|
|
// Retrieves an enrForkID from an ENR record by key lookup
|
|
// under the eth2EnrKey.
|
|
func retrieveForkEntry(record *enr.Record) (*pb.ENRForkID, error) {
|
|
sszEncodedForkEntry := make([]byte, 16)
|
|
entry := enr.WithEntry(eth2ENRKey, &sszEncodedForkEntry)
|
|
err := record.Load(entry)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
forkEntry := &pb.ENRForkID{}
|
|
if err := forkEntry.UnmarshalSSZ(sszEncodedForkEntry); err != nil {
|
|
return nil, err
|
|
}
|
|
return forkEntry, nil
|
|
}
|