erigon-pulse/cmd/sentry/sentry/sentry_grpc_server_test.go
2022-06-28 13:42:35 +02:00

179 lines
5.5 KiB
Go

package sentry
import (
"context"
"math/big"
"testing"
"time"
"github.com/holiman/uint256"
"github.com/ledgerwatch/erigon-lib/gointerfaces"
proto_sentry "github.com/ledgerwatch/erigon-lib/gointerfaces/sentry"
"github.com/ledgerwatch/erigon-lib/kv"
"github.com/ledgerwatch/erigon-lib/kv/memdb"
"github.com/ledgerwatch/erigon/common"
"github.com/ledgerwatch/erigon/core"
"github.com/ledgerwatch/erigon/core/forkid"
"github.com/ledgerwatch/erigon/core/rawdb"
"github.com/ledgerwatch/erigon/eth/protocols/eth"
"github.com/ledgerwatch/erigon/p2p"
"github.com/ledgerwatch/erigon/params"
"github.com/stretchr/testify/require"
)
func testSentryServer(db kv.Getter, genesis *core.Genesis, genesisHash common.Hash) *GrpcServer {
s := &GrpcServer{
ctx: context.Background(),
}
head := rawdb.ReadCurrentHeader(db)
headTd, err := rawdb.ReadTd(db, head.Hash(), head.Number.Uint64())
if err != nil {
panic(err)
}
headTd256 := new(uint256.Int)
headTd256.SetFromBig(headTd)
s.statusData = &proto_sentry.StatusData{
NetworkId: 1,
TotalDifficulty: gointerfaces.ConvertUint256IntToH256(headTd256),
BestHash: gointerfaces.ConvertHashToH256(head.Hash()),
MaxBlock: head.Number.Uint64(),
ForkData: &proto_sentry.Forks{
Genesis: gointerfaces.ConvertHashToH256(genesisHash),
Forks: forkid.GatherForks(genesis.Config),
},
}
return s
}
// Tests that peers are correctly accepted (or rejected) based on the advertised
// fork IDs in the protocol handshake.
func TestForkIDSplit66(t *testing.T) { testForkIDSplit(t, eth.ETH66) }
func testForkIDSplit(t *testing.T, protocol uint) {
var (
ctx = context.Background()
configNoFork = &params.ChainConfig{HomesteadBlock: big.NewInt(1), ChainID: big.NewInt(1)}
configProFork = &params.ChainConfig{
ChainID: big.NewInt(1),
HomesteadBlock: big.NewInt(1),
TangerineWhistleBlock: big.NewInt(2),
SpuriousDragonBlock: big.NewInt(2),
ByzantiumBlock: big.NewInt(3),
}
dbNoFork = memdb.NewTestDB(t)
dbProFork = memdb.NewTestDB(t)
gspecNoFork = &core.Genesis{Config: configNoFork}
gspecProFork = &core.Genesis{Config: configProFork}
genesisNoFork = gspecNoFork.MustCommit(dbNoFork)
genesisProFork = gspecProFork.MustCommit(dbProFork)
)
var s1, s2 *GrpcServer
err := dbNoFork.Update(context.Background(), func(tx kv.RwTx) error {
s1 = testSentryServer(tx, gspecNoFork, genesisNoFork.Hash())
return nil
})
require.NoError(t, err)
err = dbProFork.Update(context.Background(), func(tx kv.RwTx) error {
s2 = testSentryServer(tx, gspecProFork, genesisProFork.Hash())
return nil
})
require.NoError(t, err)
// Both nodes should allow the other to connect (same genesis, next fork is the same)
p2pNoFork, p2pProFork := p2p.MsgPipe()
defer p2pNoFork.Close()
defer p2pProFork.Close()
errc := make(chan error, 2)
go func() { errc <- handShake(ctx, s1.GetStatus(), [64]byte{1}, p2pNoFork, protocol, protocol, nil) }()
go func() { errc <- handShake(ctx, s2.GetStatus(), [64]byte{2}, p2pProFork, protocol, protocol, nil) }()
for i := 0; i < 2; i++ {
select {
case err := <-errc:
if err != nil {
t.Fatalf("frontier nofork <-> profork failed: %v", err)
}
case <-time.After(250 * time.Millisecond):
t.Fatalf("frontier nofork <-> profork handler timeout")
}
}
// Progress into Homestead. Fork's match, so we don't care what the future holds
s1.statusData.MaxBlock = 1
s2.statusData.MaxBlock = 1
go func() { errc <- handShake(ctx, s1.GetStatus(), [64]byte{1}, p2pNoFork, protocol, protocol, nil) }()
go func() { errc <- handShake(ctx, s2.GetStatus(), [64]byte{2}, p2pProFork, protocol, protocol, nil) }()
for i := 0; i < 2; i++ {
select {
case err := <-errc:
if err != nil {
t.Fatalf("homestead nofork <-> profork failed: %v", err)
}
case <-time.After(250 * time.Millisecond):
t.Fatalf("frontier nofork <-> profork handler timeout")
}
}
// Progress into Spurious. Forks mismatch, signalling differing chains, reject
s1.statusData.MaxBlock = 2
s2.statusData.MaxBlock = 2
// Both nodes should allow the other to connect (same genesis, next fork is the same)
go func() { errc <- handShake(ctx, s1.GetStatus(), [64]byte{1}, p2pNoFork, protocol, protocol, nil) }()
go func() { errc <- handShake(ctx, s2.GetStatus(), [64]byte{2}, p2pProFork, protocol, protocol, nil) }()
var successes int
for i := 0; i < 2; i++ {
select {
case err := <-errc:
if err == nil {
successes++
if successes == 2 { // Only one side disconnects
t.Fatalf("fork ID rejection didn't happen")
}
}
case <-time.After(250 * time.Millisecond):
t.Fatalf("split peers not rejected")
}
}
}
func TestSentryServerImpl_SetStatusInitPanic(t *testing.T) {
defer func() {
if r := recover(); r != nil {
t.Fatalf("panic during server initialization")
}
}()
configNoFork := &params.ChainConfig{HomesteadBlock: big.NewInt(1), ChainID: big.NewInt(1)}
dbNoFork := memdb.NewTestDB(t)
gspecNoFork := &core.Genesis{Config: configNoFork}
genesisNoFork := gspecNoFork.MustCommit(dbNoFork)
ss := &GrpcServer{p2p: &p2p.Config{}}
_, err := ss.SetStatus(context.Background(), &proto_sentry.StatusData{
ForkData: &proto_sentry.Forks{Genesis: gointerfaces.ConvertHashToH256(genesisNoFork.Hash())},
})
if err == nil {
t.Fatalf("error expected")
}
// Should not panic here.
_, err = ss.SetStatus(context.Background(), &proto_sentry.StatusData{
ForkData: &proto_sentry.Forks{Genesis: gointerfaces.ConvertHashToH256(genesisNoFork.Hash())},
})
if err == nil {
t.Fatalf("error expected")
}
}