mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-06 17:52:18 +00:00
65f71b3a48
* `subscribeStaticWithSubnets`: Fix docstring. * `buildOptions`: Avoid `options` mutations. * `dv5Cfg`: Avoid mutation. * `RefreshENR`: Use default for all but Phase0. * `udp4`, `udp6`: Create enum. * `p2p.Config`: `BootstrapNodeAddr`==> `BootstrapNodeAddrs`. * `p2p.Config`: `Discv5BootStrapAddr` ==> `Discv5BootStrapAddrs`. * `TestScorers_BadResponses_Score`: Improve. * `BeaconNode`: Avoid mutation. * `TestStore_TrustedPeers`: Remove blankline. * Remove blank identifiers. * `privKey`: Keep the majority of code with low indentation. * `P2PPreregistration`: Return error instead of fatal log. * `parseBootStrapAddrs` => `ParseBootStrapAddrs` (export) * `p2p.Config`: Remove `BootstrapNodeAddrs`. * `NewService`: Avoid mutation when possible. * `Service`: Remove blank identifier. * `buildOptions`: Avoid `log.Fatalf` (make deepsource happy). * `registerGRPCGateway`: Use `net.JoinHostPort` (make deepsource happy). * `registerBuilderService`: Make deepsource happy. * `scorers`: Add `NoLock` suffix (make deepsource happy). * `scorerr`: Add some `NoLock`suffixes (making deepsource happy). * `discovery_test.go`. Remove init. Rationale: `rand.Seed` is deprecated: As of Go 1.20 there is no reason to call Seed with a random value. Programs that call Seed with a known value to get a specific sequence of results should use New(NewSource(seed)) to obtain a local random generator. This makes deepsource happy as well. * `createListener`: Reduce cyclomatic complexity (make deepsource happy). * `startDB`: Reduce cyclomatic complexity (make deepsource happy). * `main`: Log a FATAL on error. This way, the error message is very readable. Before this commit, the error message is the less readable message in the logs. * `New`: Reduce cyclomatic complexity (make deepsource happy). * `main`: Avoid `App` mutation, and make deepsource happy. * Update beacon-chain/node/node.go Co-authored-by: Sammy Rosso <15244892+saolyn@users.noreply.github.com> * `bootnodes` ==> `BootNodes` (Fix PR comment). * Remove duplicate `configureFastSSZHashingAlgorithm` since already done in `configureBeacon`. (Fix PR comment) * Add `TestCreateLocalNode`. (PR comment fix.) * `startModules` ==> `startBaseServices (Fix PR comment). * `buildOptions` return errors consistently. * `New`: Change ordering. --------- Co-authored-by: Sammy Rosso <15244892+saolyn@users.noreply.github.com>
279 lines
9.0 KiB
Go
279 lines
9.0 KiB
Go
package p2p
|
|
|
|
import (
|
|
"context"
|
|
"math/rand"
|
|
"os"
|
|
"path"
|
|
"strconv"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/ethereum/go-ethereum/p2p/discover"
|
|
"github.com/ethereum/go-ethereum/p2p/enode"
|
|
"github.com/ethereum/go-ethereum/p2p/enr"
|
|
ma "github.com/multiformats/go-multiaddr"
|
|
mock "github.com/prysmaticlabs/prysm/v5/beacon-chain/blockchain/testing"
|
|
"github.com/prysmaticlabs/prysm/v5/beacon-chain/core/signing"
|
|
fieldparams "github.com/prysmaticlabs/prysm/v5/config/fieldparams"
|
|
"github.com/prysmaticlabs/prysm/v5/config/params"
|
|
"github.com/prysmaticlabs/prysm/v5/consensus-types/primitives"
|
|
"github.com/prysmaticlabs/prysm/v5/encoding/bytesutil"
|
|
"github.com/prysmaticlabs/prysm/v5/network/forks"
|
|
pb "github.com/prysmaticlabs/prysm/v5/proto/prysm/v1alpha1"
|
|
"github.com/prysmaticlabs/prysm/v5/testing/assert"
|
|
"github.com/prysmaticlabs/prysm/v5/testing/require"
|
|
"github.com/sirupsen/logrus"
|
|
logTest "github.com/sirupsen/logrus/hooks/test"
|
|
)
|
|
|
|
func TestStartDiscv5_DifferentForkDigests(t *testing.T) {
|
|
port := 2000
|
|
ipAddr, pkey := createAddrAndPrivKey(t)
|
|
genesisTime := time.Now()
|
|
genesisValidatorsRoot := make([]byte, fieldparams.RootLength)
|
|
s := &Service{
|
|
cfg: &Config{
|
|
UDPPort: uint(port),
|
|
StateNotifier: &mock.MockStateNotifier{},
|
|
},
|
|
genesisTime: genesisTime,
|
|
genesisValidatorsRoot: genesisValidatorsRoot,
|
|
}
|
|
bootListener, err := s.createListener(ipAddr, pkey)
|
|
require.NoError(t, err)
|
|
defer bootListener.Close()
|
|
|
|
bootNode := bootListener.Self()
|
|
cfg := &Config{
|
|
Discv5BootStrapAddrs: []string{bootNode.String()},
|
|
UDPPort: uint(port),
|
|
StateNotifier: &mock.MockStateNotifier{},
|
|
}
|
|
|
|
var listeners []*discover.UDPv5
|
|
for i := 1; i <= 5; i++ {
|
|
port = 3000 + i
|
|
cfg.UDPPort = uint(port)
|
|
ipAddr, pkey := createAddrAndPrivKey(t)
|
|
|
|
// We give every peer a different genesis validators root, which
|
|
// will cause each peer to have a different ForkDigest, preventing
|
|
// them from connecting according to our discovery rules for Ethereum consensus.
|
|
root := make([]byte, 32)
|
|
copy(root, strconv.Itoa(port))
|
|
s = &Service{
|
|
cfg: cfg,
|
|
genesisTime: genesisTime,
|
|
genesisValidatorsRoot: root,
|
|
}
|
|
listener, err := s.startDiscoveryV5(ipAddr, pkey)
|
|
assert.NoError(t, err, "Could not start discovery for node")
|
|
listeners = append(listeners, listener)
|
|
}
|
|
defer func() {
|
|
// Close down all peers.
|
|
for _, listener := range listeners {
|
|
listener.Close()
|
|
}
|
|
}()
|
|
|
|
// Wait for the nodes to have their local routing tables to be populated with the other nodes
|
|
time.Sleep(discoveryWaitTime)
|
|
|
|
lastListener := listeners[len(listeners)-1]
|
|
nodes := lastListener.Lookup(bootNode.ID())
|
|
if len(nodes) < 4 {
|
|
t.Errorf("The node's local table doesn't have the expected number of nodes. "+
|
|
"Expected more than or equal to %d but got %d", 4, len(nodes))
|
|
}
|
|
|
|
// Now, we start a new p2p service. It should have no peers aside from the
|
|
// bootnode given all nodes provided by discv5 will have different fork digests.
|
|
cfg.UDPPort = 14000
|
|
cfg.TCPPort = 14001
|
|
cfg.MaxPeers = 30
|
|
s, err = NewService(context.Background(), cfg)
|
|
require.NoError(t, err)
|
|
s.genesisTime = genesisTime
|
|
s.genesisValidatorsRoot = make([]byte, 32)
|
|
s.dv5Listener = lastListener
|
|
var addrs []ma.Multiaddr
|
|
|
|
for _, n := range nodes {
|
|
if s.filterPeer(n) {
|
|
addr, err := convertToSingleMultiAddr(n)
|
|
require.NoError(t, err)
|
|
addrs = append(addrs, addr)
|
|
}
|
|
}
|
|
|
|
// We should not have valid peers if the fork digest mismatched.
|
|
assert.Equal(t, 0, len(addrs), "Expected 0 valid peers")
|
|
require.NoError(t, s.Stop())
|
|
}
|
|
|
|
func TestStartDiscv5_SameForkDigests_DifferentNextForkData(t *testing.T) {
|
|
params.SetupTestConfigCleanup(t)
|
|
hook := logTest.NewGlobal()
|
|
logrus.SetLevel(logrus.TraceLevel)
|
|
port := 2000
|
|
ipAddr, pkey := createAddrAndPrivKey(t)
|
|
genesisTime := time.Now()
|
|
genesisValidatorsRoot := make([]byte, 32)
|
|
s := &Service{
|
|
cfg: &Config{UDPPort: uint(port)},
|
|
genesisTime: genesisTime,
|
|
genesisValidatorsRoot: genesisValidatorsRoot,
|
|
}
|
|
bootListener, err := s.createListener(ipAddr, pkey)
|
|
require.NoError(t, err)
|
|
defer bootListener.Close()
|
|
|
|
bootNode := bootListener.Self()
|
|
cfg := &Config{
|
|
Discv5BootStrapAddrs: []string{bootNode.String()},
|
|
UDPPort: uint(port),
|
|
}
|
|
|
|
var listeners []*discover.UDPv5
|
|
for i := 1; i <= 5; i++ {
|
|
port = 3000 + i
|
|
cfg.UDPPort = uint(port)
|
|
ipAddr, pkey := createAddrAndPrivKey(t)
|
|
|
|
c := params.BeaconConfig().Copy()
|
|
nextForkEpoch := primitives.Epoch(i)
|
|
c.ForkVersionSchedule[[4]byte{'A', 'B', 'C', 'D'}] = nextForkEpoch
|
|
params.OverrideBeaconConfig(c)
|
|
|
|
// We give every peer a different genesis validators root, which
|
|
// will cause each peer to have a different ForkDigest, preventing
|
|
// them from connecting according to our discovery rules for Ethereum consensus.
|
|
s = &Service{
|
|
cfg: cfg,
|
|
genesisTime: genesisTime,
|
|
genesisValidatorsRoot: genesisValidatorsRoot,
|
|
}
|
|
listener, err := s.startDiscoveryV5(ipAddr, pkey)
|
|
assert.NoError(t, err, "Could not start discovery for node")
|
|
listeners = append(listeners, listener)
|
|
}
|
|
defer func() {
|
|
// Close down all peers.
|
|
for _, listener := range listeners {
|
|
listener.Close()
|
|
}
|
|
}()
|
|
|
|
// Wait for the nodes to have their local routing tables to be populated with the other nodes
|
|
time.Sleep(discoveryWaitTime)
|
|
|
|
lastListener := listeners[len(listeners)-1]
|
|
nodes := lastListener.Lookup(bootNode.ID())
|
|
if len(nodes) < 4 {
|
|
t.Errorf("The node's local table doesn't have the expected number of nodes. "+
|
|
"Expected more than or equal to %d but got %d", 4, len(nodes))
|
|
}
|
|
|
|
// Now, we start a new p2p service. It should have no peers aside from the
|
|
// bootnode given all nodes provided by discv5 will have different fork digests.
|
|
cfg.UDPPort = 14000
|
|
cfg.TCPPort = 14001
|
|
cfg.MaxPeers = 30
|
|
cfg.StateNotifier = &mock.MockStateNotifier{}
|
|
s, err = NewService(context.Background(), cfg)
|
|
require.NoError(t, err)
|
|
|
|
s.genesisTime = genesisTime
|
|
s.genesisValidatorsRoot = make([]byte, 32)
|
|
s.dv5Listener = lastListener
|
|
var addrs []ma.Multiaddr
|
|
|
|
for _, n := range nodes {
|
|
if s.filterPeer(n) {
|
|
addr, err := convertToSingleMultiAddr(n)
|
|
require.NoError(t, err)
|
|
addrs = append(addrs, addr)
|
|
}
|
|
}
|
|
if len(addrs) == 0 {
|
|
t.Error("Expected to have valid peers, got 0")
|
|
}
|
|
|
|
require.LogsContain(t, hook, "Peer matches fork digest but has different next fork epoch")
|
|
require.NoError(t, s.Stop())
|
|
}
|
|
|
|
func TestDiscv5_AddRetrieveForkEntryENR(t *testing.T) {
|
|
params.SetupTestConfigCleanup(t)
|
|
c := params.BeaconConfig().Copy()
|
|
c.ForkVersionSchedule = map[[4]byte]primitives.Epoch{
|
|
bytesutil.ToBytes4(params.BeaconConfig().GenesisForkVersion): 0,
|
|
{0, 0, 0, 1}: 1,
|
|
}
|
|
nextForkEpoch := primitives.Epoch(1)
|
|
nextForkVersion := []byte{0, 0, 0, 1}
|
|
params.OverrideBeaconConfig(c)
|
|
|
|
genesisTime := time.Now()
|
|
genesisValidatorsRoot := make([]byte, 32)
|
|
digest, err := forks.CreateForkDigest(genesisTime, make([]byte, 32))
|
|
require.NoError(t, err)
|
|
enrForkID := &pb.ENRForkID{
|
|
CurrentForkDigest: digest[:],
|
|
NextForkVersion: nextForkVersion,
|
|
NextForkEpoch: nextForkEpoch,
|
|
}
|
|
enc, err := enrForkID.MarshalSSZ()
|
|
require.NoError(t, err)
|
|
entry := enr.WithEntry(eth2ENRKey, enc)
|
|
// In epoch 1 of current time, the fork version should be
|
|
// {0, 0, 0, 1} according to the configuration override above.
|
|
temp := t.TempDir()
|
|
randNum := rand.Int()
|
|
tempPath := path.Join(temp, strconv.Itoa(randNum))
|
|
require.NoError(t, os.Mkdir(tempPath, 0700))
|
|
pkey, err := privKey(&Config{DataDir: tempPath})
|
|
require.NoError(t, err, "Could not get private key")
|
|
db, err := enode.OpenDB("")
|
|
require.NoError(t, err)
|
|
localNode := enode.NewLocalNode(db, pkey)
|
|
localNode.Set(entry)
|
|
|
|
want, err := signing.ComputeForkDigest([]byte{0, 0, 0, 0}, genesisValidatorsRoot)
|
|
require.NoError(t, err)
|
|
|
|
resp, err := forkEntry(localNode.Node().Record())
|
|
require.NoError(t, err)
|
|
assert.DeepEqual(t, want[:], resp.CurrentForkDigest)
|
|
assert.DeepEqual(t, nextForkVersion, resp.NextForkVersion)
|
|
assert.Equal(t, nextForkEpoch, resp.NextForkEpoch, "Unexpected next fork epoch")
|
|
}
|
|
|
|
func TestAddForkEntry_Genesis(t *testing.T) {
|
|
params.SetupTestConfigCleanup(t)
|
|
temp := t.TempDir()
|
|
randNum := rand.Int()
|
|
tempPath := path.Join(temp, strconv.Itoa(randNum))
|
|
require.NoError(t, os.Mkdir(tempPath, 0700))
|
|
pkey, err := privKey(&Config{DataDir: tempPath})
|
|
require.NoError(t, err, "Could not get private key")
|
|
db, err := enode.OpenDB("")
|
|
require.NoError(t, err)
|
|
|
|
bCfg := params.MainnetConfig().Copy()
|
|
bCfg.ForkVersionSchedule = map[[4]byte]primitives.Epoch{}
|
|
bCfg.ForkVersionSchedule[bytesutil.ToBytes4(params.BeaconConfig().GenesisForkVersion)] = bCfg.GenesisEpoch
|
|
params.OverrideBeaconConfig(bCfg)
|
|
|
|
localNode := enode.NewLocalNode(db, pkey)
|
|
localNode, err = addForkEntry(localNode, time.Now().Add(10*time.Second), bytesutil.PadTo([]byte{'A', 'B', 'C', 'D'}, 32))
|
|
require.NoError(t, err)
|
|
forkEntry, err := forkEntry(localNode.Node().Record())
|
|
require.NoError(t, err)
|
|
assert.DeepEqual(t,
|
|
params.BeaconConfig().GenesisForkVersion, forkEntry.NextForkVersion,
|
|
"Wanted Next Fork Version to be equal to genesis fork version")
|
|
}
|