package beacon import ( "context" "encoding/binary" "testing" "time" "github.com/gogo/protobuf/proto" ethpb "github.com/prysmaticlabs/ethereumapis/eth/v1alpha1" mock "github.com/prysmaticlabs/prysm/beacon-chain/blockchain/testing" "github.com/prysmaticlabs/prysm/beacon-chain/core/helpers" dbTest "github.com/prysmaticlabs/prysm/beacon-chain/db/testing" iface "github.com/prysmaticlabs/prysm/beacon-chain/state/interface" "github.com/prysmaticlabs/prysm/beacon-chain/state/stategen" pbp2p "github.com/prysmaticlabs/prysm/proto/beacon/p2p/v1" "github.com/prysmaticlabs/prysm/shared/params" "github.com/prysmaticlabs/prysm/shared/testutil" "github.com/prysmaticlabs/prysm/shared/testutil/assert" "github.com/prysmaticlabs/prysm/shared/testutil/require" "github.com/prysmaticlabs/prysm/shared/timeutils" "gopkg.in/d4l3k/messagediff.v1" ) func TestServer_ListBeaconCommittees_CurrentEpoch(t *testing.T) { db := dbTest.SetupDB(t) helpers.ClearCache() numValidators := 128 ctx := context.Background() headState := setupActiveValidators(t, numValidators) offset := int64(headState.Slot().Mul(params.BeaconConfig().SecondsPerSlot)) m := &mock.ChainService{ Genesis: timeutils.Now().Add(time.Duration(-1*offset) * time.Second), } bs := &Server{ HeadFetcher: m, GenesisTimeFetcher: m, StateGen: stategen.New(db), } b := testutil.NewBeaconBlock() require.NoError(t, db.SaveBlock(ctx, b)) gRoot, err := b.Block.HashTreeRoot() require.NoError(t, err) require.NoError(t, db.SaveGenesisBlockRoot(ctx, gRoot)) require.NoError(t, db.SaveState(ctx, headState, gRoot)) activeIndices, err := helpers.ActiveValidatorIndices(headState, 0) require.NoError(t, err) attesterSeed, err := helpers.Seed(headState, 0, params.BeaconConfig().DomainBeaconAttester) require.NoError(t, err) committees, err := computeCommittees(0, activeIndices, attesterSeed) require.NoError(t, err) wanted := ðpb.BeaconCommittees{ Epoch: 0, Committees: committees.SlotToUint64(), ActiveValidatorCount: uint64(numValidators), } res, err := bs.ListBeaconCommittees(context.Background(), ðpb.ListCommitteesRequest{ QueryFilter: ðpb.ListCommitteesRequest_Genesis{Genesis: true}, }) require.NoError(t, err) if !proto.Equal(res, wanted) { t.Errorf("Expected %v, received %v", wanted, res) } } func TestServer_ListBeaconCommittees_PreviousEpoch(t *testing.T) { params.UseMainnetConfig() ctx := context.Background() db := dbTest.SetupDB(t) helpers.ClearCache() numValidators := 128 headState := setupActiveValidators(t, numValidators) mixes := make([][]byte, params.BeaconConfig().EpochsPerHistoricalVector) for i := 0; i < len(mixes); i++ { mixes[i] = make([]byte, 32) } require.NoError(t, headState.SetRandaoMixes(mixes)) require.NoError(t, headState.SetSlot(params.BeaconConfig().SlotsPerEpoch)) b := testutil.NewBeaconBlock() require.NoError(t, db.SaveBlock(ctx, b)) gRoot, err := b.Block.HashTreeRoot() require.NoError(t, err) require.NoError(t, db.SaveState(ctx, headState, gRoot)) require.NoError(t, db.SaveGenesisBlockRoot(ctx, gRoot)) offset := int64(headState.Slot().Mul(params.BeaconConfig().SecondsPerSlot)) m := &mock.ChainService{ State: headState, Genesis: timeutils.Now().Add(time.Duration(-1*offset) * time.Second), } bs := &Server{ HeadFetcher: m, GenesisTimeFetcher: m, StateGen: stategen.New(db), } activeIndices, err := helpers.ActiveValidatorIndices(headState, 1) require.NoError(t, err) attesterSeed, err := helpers.Seed(headState, 1, params.BeaconConfig().DomainBeaconAttester) require.NoError(t, err) startSlot, err := helpers.StartSlot(1) require.NoError(t, err) wanted, err := computeCommittees(startSlot, activeIndices, attesterSeed) require.NoError(t, err) tests := []struct { req *ethpb.ListCommitteesRequest res *ethpb.BeaconCommittees }{ { req: ðpb.ListCommitteesRequest{ QueryFilter: ðpb.ListCommitteesRequest_Epoch{Epoch: 1}, }, res: ðpb.BeaconCommittees{ Epoch: 1, Committees: wanted.SlotToUint64(), ActiveValidatorCount: uint64(numValidators), }, }, } helpers.ClearCache() for i, test := range tests { res, err := bs.ListBeaconCommittees(context.Background(), test.req) require.NoError(t, err) if !proto.Equal(res, test.res) { diff, _ := messagediff.PrettyDiff(res, test.res) t.Errorf("%d/ Diff between responses %s", i, diff) } } } func TestRetrieveCommitteesForRoot(t *testing.T) { db := dbTest.SetupDB(t) helpers.ClearCache() ctx := context.Background() numValidators := 128 headState := setupActiveValidators(t, numValidators) offset := int64(headState.Slot().Mul(params.BeaconConfig().SecondsPerSlot)) m := &mock.ChainService{ Genesis: timeutils.Now().Add(time.Duration(-1*offset) * time.Second), } bs := &Server{ HeadFetcher: m, GenesisTimeFetcher: m, StateGen: stategen.New(db), } b := testutil.NewBeaconBlock() require.NoError(t, db.SaveBlock(ctx, b)) gRoot, err := b.Block.HashTreeRoot() require.NoError(t, err) require.NoError(t, db.SaveGenesisBlockRoot(ctx, gRoot)) require.NoError(t, db.SaveState(ctx, headState, gRoot)) stateSummary := &pbp2p.StateSummary{ Slot: 0, Root: gRoot[:], } require.NoError(t, db.SaveStateSummary(ctx, stateSummary)) // Store the genesis seed. seed, err := helpers.Seed(headState, 0, params.BeaconConfig().DomainBeaconAttester) require.NoError(t, err) require.NoError(t, headState.SetSlot(params.BeaconConfig().SlotsPerEpoch*10)) activeIndices, err := helpers.ActiveValidatorIndices(headState, 0) require.NoError(t, err) wanted, err := computeCommittees(0, activeIndices, seed) require.NoError(t, err) committees, activeIndices, err := bs.retrieveCommitteesForRoot(context.Background(), gRoot[:]) require.NoError(t, err) wantedRes := ðpb.BeaconCommittees{ Epoch: 0, Committees: wanted.SlotToUint64(), ActiveValidatorCount: uint64(numValidators), } receivedRes := ðpb.BeaconCommittees{ Epoch: 0, Committees: committees.SlotToUint64(), ActiveValidatorCount: uint64(len(activeIndices)), } assert.DeepEqual(t, wantedRes, receivedRes) } func setupActiveValidators(t *testing.T, count int) iface.BeaconState { balances := make([]uint64, count) validators := make([]*ethpb.Validator, 0, count) for i := 0; i < count; i++ { pubKey := make([]byte, params.BeaconConfig().BLSPubkeyLength) binary.LittleEndian.PutUint64(pubKey, uint64(i)) balances[i] = uint64(i) validators = append(validators, ðpb.Validator{ PublicKey: pubKey, ActivationEpoch: 0, ExitEpoch: params.BeaconConfig().FarFutureEpoch, WithdrawalCredentials: make([]byte, 32), }) } s, err := testutil.NewBeaconState() require.NoError(t, err) if err := s.SetValidators(validators); err != nil { t.Error(err) return nil } if err := s.SetBalances(balances); err != nil { t.Error(err) return nil } return s }