prysm-pulse/beacon-chain/rpc/prysm/v1alpha1/validator/server_test.go
kasey 918129cf36
Replace statefeed Initialize (#12285)
* refactor initialization to blocking startup method

* require genesisSetter in blockchain, fix tests

* work-around gazelle weirdness

* fix dep gazelle ignores

* only call SetGenesis once

* fix typo

* validator test setup and fix to return right error

* move waitForChainStart to Start

* wire up sync Service.genesisWaiter

* fix p2p genesisWaiter plumbing

* remove extra clock type, integrate into genesis

and rename

* use time.Now when no Nower is specified

* remove unused ClockSetter

* simplify rpc context checking

* fix typo

* use clock everywhere in sync; [32]byte val root

* don't use DeepEqual to compare [32]byte and []byte

* don't use clock in init sync, not wired up yet

* use clock waiter in blockchain as well

* use cancelable contexts in tests with goroutines

* missed a reference to WithClockSetter

* Update beacon-chain/startup/genesis.go

Co-authored-by: Radosław Kapka <rkapka@wp.pl>

* Update beacon-chain/blockchain/service_test.go

Co-authored-by: Radosław Kapka <rkapka@wp.pl>

* more clear docs

* doc for NewClock

* move clock typedef to more logical file name

* adding documentation

* gaz

* fixes for capella

* reducing test raciness

* fix races in committee cache tests

* lint

* add tests on Duration slot math helper

* startup package test coverage

* fix bad merge

* set non-zero genesis time in tests that call Start

* happy deepsource, happy me-epsource

* replace Synced event with channel

* remove unused error

* remove accidental wip commit

* gaz!

* remove unused event constants

* remove sync statefeed subscription to fix deadlock

* remove state notifier

* fix build

---------

Co-authored-by: Kasey Kirkham <kasey@users.noreply.github.com>
Co-authored-by: Radosław Kapka <rkapka@wp.pl>
Co-authored-by: prylabs-bulldozer[bot] <58059840+prylabs-bulldozer[bot]@users.noreply.github.com>
Co-authored-by: nisdas <nishdas93@gmail.com>
2023-05-03 04:34:01 +00:00

312 lines
9.6 KiB
Go

package validator
import (
"context"
"sync"
"testing"
"time"
"github.com/golang/mock/gomock"
"github.com/prysmaticlabs/prysm/v4/async/event"
mockChain "github.com/prysmaticlabs/prysm/v4/beacon-chain/blockchain/testing"
"github.com/prysmaticlabs/prysm/v4/beacon-chain/cache/depositcache"
mockExecution "github.com/prysmaticlabs/prysm/v4/beacon-chain/execution/testing"
"github.com/prysmaticlabs/prysm/v4/beacon-chain/startup"
state_native "github.com/prysmaticlabs/prysm/v4/beacon-chain/state/state-native"
"github.com/prysmaticlabs/prysm/v4/config/params"
"github.com/prysmaticlabs/prysm/v4/crypto/bls"
"github.com/prysmaticlabs/prysm/v4/encoding/bytesutil"
ethpb "github.com/prysmaticlabs/prysm/v4/proto/prysm/v1alpha1"
"github.com/prysmaticlabs/prysm/v4/testing/assert"
"github.com/prysmaticlabs/prysm/v4/testing/mock"
"github.com/prysmaticlabs/prysm/v4/testing/require"
"github.com/prysmaticlabs/prysm/v4/testing/util"
logTest "github.com/sirupsen/logrus/hooks/test"
"google.golang.org/protobuf/types/known/emptypb"
)
func TestValidatorIndex_OK(t *testing.T) {
st, err := util.NewBeaconState()
require.NoError(t, err)
pubKey := pubKey(1)
err = st.SetValidators([]*ethpb.Validator{{PublicKey: pubKey}})
require.NoError(t, err)
Server := &Server{
HeadFetcher: &mockChain.ChainService{State: st},
}
req := &ethpb.ValidatorIndexRequest{
PublicKey: pubKey,
}
_, err = Server.ValidatorIndex(context.Background(), req)
assert.NoError(t, err, "Could not get validator index")
}
func TestValidatorIndex_StateEmpty(t *testing.T) {
Server := &Server{
HeadFetcher: &mockChain.ChainService{},
}
pubKey := pubKey(1)
req := &ethpb.ValidatorIndexRequest{
PublicKey: pubKey,
}
_, err := Server.ValidatorIndex(context.Background(), req)
assert.ErrorContains(t, "head state is empty", err)
}
func TestWaitForActivation_ContextClosed(t *testing.T) {
beaconState, err := state_native.InitializeFromProtoPhase0(&ethpb.BeaconState{
Slot: 0,
Validators: []*ethpb.Validator{},
})
require.NoError(t, err)
block := util.NewBeaconBlock()
genesisRoot, err := block.Block.HashTreeRoot()
require.NoError(t, err, "Could not get signing root")
ctx, cancel := context.WithCancel(context.Background())
depositCache, err := depositcache.New()
require.NoError(t, err)
vs := &Server{
Ctx: ctx,
ChainStartFetcher: &mockExecution.Chain{},
BlockFetcher: &mockExecution.Chain{},
Eth1InfoFetcher: &mockExecution.Chain{},
DepositFetcher: depositCache,
HeadFetcher: &mockChain.ChainService{State: beaconState, Root: genesisRoot[:]},
}
req := &ethpb.ValidatorActivationRequest{
PublicKeys: [][]byte{pubKey(1)},
}
ctrl := gomock.NewController(t)
defer ctrl.Finish()
mockChainStream := mock.NewMockBeaconNodeValidator_WaitForActivationServer(ctrl)
mockChainStream.EXPECT().Context().Return(context.Background())
mockChainStream.EXPECT().Send(gomock.Any()).Return(nil)
mockChainStream.EXPECT().Context().Return(context.Background())
exitRoutine := make(chan bool)
go func(tt *testing.T) {
want := "context canceled"
assert.ErrorContains(tt, want, vs.WaitForActivation(req, mockChainStream))
<-exitRoutine
}(t)
cancel()
exitRoutine <- true
}
func TestWaitForActivation_MultipleStatuses(t *testing.T) {
priv1, err := bls.RandKey()
require.NoError(t, err)
priv2, err := bls.RandKey()
require.NoError(t, err)
priv3, err := bls.RandKey()
require.NoError(t, err)
pubKey1 := priv1.PublicKey().Marshal()
pubKey2 := priv2.PublicKey().Marshal()
pubKey3 := priv3.PublicKey().Marshal()
beaconState := &ethpb.BeaconState{
Slot: 4000,
Validators: []*ethpb.Validator{
{
PublicKey: pubKey1,
ActivationEpoch: 1,
ExitEpoch: params.BeaconConfig().FarFutureEpoch,
},
{
PublicKey: pubKey2,
ActivationEpoch: params.BeaconConfig().FarFutureEpoch,
ActivationEligibilityEpoch: 6,
ExitEpoch: params.BeaconConfig().FarFutureEpoch,
},
{
PublicKey: pubKey3,
ActivationEpoch: 0,
ActivationEligibilityEpoch: 0,
ExitEpoch: 0,
},
},
}
block := util.NewBeaconBlock()
genesisRoot, err := block.Block.HashTreeRoot()
require.NoError(t, err, "Could not get signing root")
s, err := state_native.InitializeFromProtoUnsafePhase0(beaconState)
require.NoError(t, err)
vs := &Server{
Ctx: context.Background(),
ChainStartFetcher: &mockExecution.Chain{},
HeadFetcher: &mockChain.ChainService{State: s, Root: genesisRoot[:]},
}
req := &ethpb.ValidatorActivationRequest{
PublicKeys: [][]byte{pubKey1, pubKey2, pubKey3},
}
ctrl := gomock.NewController(t)
defer ctrl.Finish()
mockChainStream := mock.NewMockBeaconNodeValidator_WaitForActivationServer(ctrl)
mockChainStream.EXPECT().Context().Return(context.Background())
mockChainStream.EXPECT().Send(
&ethpb.ValidatorActivationResponse{
Statuses: []*ethpb.ValidatorActivationResponse_Status{
{
PublicKey: pubKey1,
Status: &ethpb.ValidatorStatusResponse{
Status: ethpb.ValidatorStatus_ACTIVE,
ActivationEpoch: 1,
},
Index: 0,
},
{
PublicKey: pubKey2,
Status: &ethpb.ValidatorStatusResponse{
Status: ethpb.ValidatorStatus_PENDING,
ActivationEpoch: params.BeaconConfig().FarFutureEpoch,
PositionInActivationQueue: 1,
},
Index: 1,
},
{
PublicKey: pubKey3,
Status: &ethpb.ValidatorStatusResponse{
Status: ethpb.ValidatorStatus_EXITED,
},
Index: 2,
},
},
},
).Return(nil)
require.NoError(t, vs.WaitForActivation(req, mockChainStream), "Could not setup wait for activation stream")
}
func TestWaitForChainStart_ContextClosed(t *testing.T) {
ctx, cancel := context.WithCancel(context.Background())
chainService := &mockChain.ChainService{}
server := &Server{
Ctx: ctx,
ChainStartFetcher: &mockExecution.FaultyExecutionChain{
ChainFeed: new(event.Feed),
},
StateNotifier: chainService.StateNotifier(),
HeadFetcher: chainService,
ClockWaiter: startup.NewClockSynchronizer(),
}
exitRoutine := make(chan bool)
ctrl := gomock.NewController(t)
defer ctrl.Finish()
mockStream := mock.NewMockBeaconNodeValidator_WaitForChainStartServer(ctrl)
mockStream.EXPECT().Context().Return(ctx)
go func(tt *testing.T) {
err := server.WaitForChainStart(&emptypb.Empty{}, mockStream)
assert.ErrorContains(tt, "Context canceled", err)
<-exitRoutine
}(t)
cancel()
exitRoutine <- true
}
func TestWaitForChainStart_AlreadyStarted(t *testing.T) {
st, err := util.NewBeaconState()
require.NoError(t, err)
require.NoError(t, st.SetSlot(3))
genesisValidatorsRoot := bytesutil.ToBytes32([]byte("validators"))
require.NoError(t, st.SetGenesisValidatorsRoot(genesisValidatorsRoot[:]))
chainService := &mockChain.ChainService{State: st, ValidatorsRoot: genesisValidatorsRoot}
Server := &Server{
Ctx: context.Background(),
ChainStartFetcher: &mockExecution.Chain{
ChainFeed: new(event.Feed),
},
StateNotifier: chainService.StateNotifier(),
HeadFetcher: chainService,
}
ctrl := gomock.NewController(t)
defer ctrl.Finish()
mockStream := mock.NewMockBeaconNodeValidator_WaitForChainStartServer(ctrl)
mockStream.EXPECT().Send(
&ethpb.ChainStartResponse{
Started: true,
GenesisTime: uint64(time.Unix(0, 0).Unix()),
GenesisValidatorsRoot: genesisValidatorsRoot[:],
},
).Return(nil)
mockStream.EXPECT().Context().Return(context.Background())
assert.NoError(t, Server.WaitForChainStart(&emptypb.Empty{}, mockStream), "Could not call RPC method")
}
func TestWaitForChainStart_HeadStateDoesNotExist(t *testing.T) {
// Set head state to nil
chainService := &mockChain.ChainService{State: nil}
gs := startup.NewClockSynchronizer()
Server := &Server{
Ctx: context.Background(),
ChainStartFetcher: &mockExecution.Chain{
ChainFeed: new(event.Feed),
},
StateNotifier: chainService.StateNotifier(),
HeadFetcher: chainService,
ClockWaiter: gs,
}
ctrl := gomock.NewController(t)
defer ctrl.Finish()
mockStream := mock.NewMockBeaconNodeValidator_WaitForChainStartServer(ctrl)
mockStream.EXPECT().Context().Return(context.Background())
wg := new(sync.WaitGroup)
wg.Add(1)
go func() {
assert.NoError(t, Server.WaitForChainStart(&emptypb.Empty{}, mockStream), "Could not call RPC method")
wg.Done()
}()
util.WaitTimeout(wg, time.Second)
}
func TestWaitForChainStart_NotStartedThenLogFired(t *testing.T) {
hook := logTest.NewGlobal()
genesisValidatorsRoot := bytesutil.ToBytes32([]byte("validators"))
chainService := &mockChain.ChainService{}
gs := startup.NewClockSynchronizer()
Server := &Server{
Ctx: context.Background(),
ChainStartFetcher: &mockExecution.FaultyExecutionChain{
ChainFeed: new(event.Feed),
},
StateNotifier: chainService.StateNotifier(),
HeadFetcher: chainService,
ClockWaiter: gs,
}
exitRoutine := make(chan bool)
ctrl := gomock.NewController(t)
defer ctrl.Finish()
mockStream := mock.NewMockBeaconNodeValidator_WaitForChainStartServer(ctrl)
mockStream.EXPECT().Send(
&ethpb.ChainStartResponse{
Started: true,
GenesisTime: uint64(time.Unix(0, 0).Unix()),
GenesisValidatorsRoot: genesisValidatorsRoot[:],
},
).Return(nil)
mockStream.EXPECT().Context().Return(context.Background())
go func(tt *testing.T) {
assert.NoError(tt, Server.WaitForChainStart(&emptypb.Empty{}, mockStream))
<-exitRoutine
}(t)
// Send in a loop to ensure it is delivered (busy wait for the service to subscribe to the state feed).
require.NoError(t, gs.SetClock(startup.NewClock(time.Unix(0, 0), genesisValidatorsRoot)))
exitRoutine <- true
require.LogsContain(t, hook, "Sending genesis time")
}