mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-17 23:38:46 +00:00
943dec525c
* Delete deploy contract tool. Move mock to its own package as testonly with some helper functions * gofmt contracts/deposit/mock/mock.go * move stategen mock.go to its on testonly pkg * move password_reader_mock.go to mock testonly package * move mock keymanager to its own testonly package * move attestations mock * move voluntaryexits mock * Move slashings mock to mock pkg * move the slasher mock Co-authored-by: prylabs-bulldozer[bot] <58059840+prylabs-bulldozer[bot]@users.noreply.github.com>
1319 lines
44 KiB
Go
1319 lines
44 KiB
Go
package validator
|
|
|
|
import (
|
|
"context"
|
|
"reflect"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/d4l3k/messagediff"
|
|
types "github.com/prysmaticlabs/eth2-types"
|
|
"github.com/prysmaticlabs/go-bitfield"
|
|
mockChain "github.com/prysmaticlabs/prysm/beacon-chain/blockchain/testing"
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/cache/depositcache"
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/core/helpers"
|
|
mockPOW "github.com/prysmaticlabs/prysm/beacon-chain/powchain/testing"
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/state"
|
|
mockstategen "github.com/prysmaticlabs/prysm/beacon-chain/state/stategen/mock"
|
|
v1 "github.com/prysmaticlabs/prysm/beacon-chain/state/v1"
|
|
mockSync "github.com/prysmaticlabs/prysm/beacon-chain/sync/initial-sync/testing"
|
|
fieldparams "github.com/prysmaticlabs/prysm/config/fieldparams"
|
|
"github.com/prysmaticlabs/prysm/config/params"
|
|
"github.com/prysmaticlabs/prysm/container/trie"
|
|
"github.com/prysmaticlabs/prysm/crypto/bls"
|
|
"github.com/prysmaticlabs/prysm/encoding/bytesutil"
|
|
ethpb "github.com/prysmaticlabs/prysm/proto/prysm/v1alpha1"
|
|
"github.com/prysmaticlabs/prysm/testing/assert"
|
|
"github.com/prysmaticlabs/prysm/testing/require"
|
|
"github.com/prysmaticlabs/prysm/testing/util"
|
|
"github.com/prysmaticlabs/prysm/time/slots"
|
|
"google.golang.org/protobuf/proto"
|
|
)
|
|
|
|
func TestValidatorStatus_DepositedEth1(t *testing.T) {
|
|
ctx := context.Background()
|
|
deposits, _, err := util.DeterministicDepositsAndKeys(1)
|
|
require.NoError(t, err, "Could not generate deposits and keys")
|
|
deposit := deposits[0]
|
|
pubKey1 := deposit.Data.PublicKey
|
|
depositTrie, err := trie.NewTrie(params.BeaconConfig().DepositContractTreeDepth)
|
|
require.NoError(t, err, "Could not setup deposit trie")
|
|
depositCache, err := depositcache.New()
|
|
require.NoError(t, err)
|
|
|
|
assert.NoError(t, depositCache.InsertDeposit(ctx, deposit, 0 /*blockNum*/, 0, depositTrie.HashTreeRoot()))
|
|
height := time.Unix(int64(params.BeaconConfig().Eth1FollowDistance), 0).Unix()
|
|
p := &mockPOW.POWChain{
|
|
TimesByHeight: map[int]uint64{
|
|
0: uint64(height),
|
|
},
|
|
}
|
|
stateObj, err := v1.InitializeFromProtoUnsafe(ðpb.BeaconState{})
|
|
require.NoError(t, err)
|
|
vs := &Server{
|
|
DepositFetcher: depositCache,
|
|
BlockFetcher: p,
|
|
HeadFetcher: &mockChain.ChainService{
|
|
State: stateObj,
|
|
},
|
|
Eth1InfoFetcher: p,
|
|
}
|
|
req := ðpb.ValidatorStatusRequest{
|
|
PublicKey: pubKey1,
|
|
}
|
|
resp, err := vs.ValidatorStatus(context.Background(), req)
|
|
require.NoError(t, err, "Could not get validator status")
|
|
assert.Equal(t, ethpb.ValidatorStatus_DEPOSITED, resp.Status)
|
|
}
|
|
|
|
func TestValidatorStatus_Deposited(t *testing.T) {
|
|
ctx := context.Background()
|
|
|
|
pubKey1 := pubKey(1)
|
|
depData := ðpb.Deposit_Data{
|
|
Amount: params.BeaconConfig().MaxEffectiveBalance,
|
|
PublicKey: pubKey1,
|
|
Signature: bytesutil.PadTo([]byte("hi"), 96),
|
|
WithdrawalCredentials: bytesutil.PadTo([]byte("hey"), 32),
|
|
}
|
|
deposit := ðpb.Deposit{
|
|
Data: depData,
|
|
}
|
|
depositTrie, err := trie.NewTrie(params.BeaconConfig().DepositContractTreeDepth)
|
|
require.NoError(t, err, "Could not setup deposit trie")
|
|
depositCache, err := depositcache.New()
|
|
require.NoError(t, err)
|
|
|
|
assert.NoError(t, depositCache.InsertDeposit(ctx, deposit, 0 /*blockNum*/, 0, depositTrie.HashTreeRoot()))
|
|
height := time.Unix(int64(params.BeaconConfig().Eth1FollowDistance), 0).Unix()
|
|
p := &mockPOW.POWChain{
|
|
TimesByHeight: map[int]uint64{
|
|
0: uint64(height),
|
|
},
|
|
}
|
|
stateObj, err := v1.InitializeFromProtoUnsafe(ðpb.BeaconState{
|
|
Validators: []*ethpb.Validator{
|
|
{
|
|
PublicKey: pubKey1,
|
|
ActivationEligibilityEpoch: 1,
|
|
},
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
vs := &Server{
|
|
DepositFetcher: depositCache,
|
|
BlockFetcher: p,
|
|
HeadFetcher: &mockChain.ChainService{
|
|
State: stateObj,
|
|
},
|
|
Eth1InfoFetcher: p,
|
|
}
|
|
req := ðpb.ValidatorStatusRequest{
|
|
PublicKey: pubKey1,
|
|
}
|
|
resp, err := vs.ValidatorStatus(context.Background(), req)
|
|
require.NoError(t, err, "Could not get validator status")
|
|
assert.Equal(t, ethpb.ValidatorStatus_DEPOSITED, resp.Status)
|
|
}
|
|
|
|
func TestValidatorStatus_PartiallyDeposited(t *testing.T) {
|
|
ctx := context.Background()
|
|
|
|
pubKey1 := pubKey(1)
|
|
depData := ðpb.Deposit_Data{
|
|
Amount: params.BeaconConfig().MinDepositAmount,
|
|
PublicKey: pubKey1,
|
|
Signature: []byte("hi"),
|
|
WithdrawalCredentials: []byte("hey"),
|
|
}
|
|
deposit := ðpb.Deposit{
|
|
Data: depData,
|
|
}
|
|
depositTrie, err := trie.NewTrie(params.BeaconConfig().DepositContractTreeDepth)
|
|
require.NoError(t, err, "Could not setup deposit trie")
|
|
depositCache, err := depositcache.New()
|
|
require.NoError(t, err)
|
|
|
|
assert.NoError(t, depositCache.InsertDeposit(ctx, deposit, 0 /*blockNum*/, 0, depositTrie.HashTreeRoot()))
|
|
height := time.Unix(int64(params.BeaconConfig().Eth1FollowDistance), 0).Unix()
|
|
p := &mockPOW.POWChain{
|
|
TimesByHeight: map[int]uint64{
|
|
0: uint64(height),
|
|
},
|
|
}
|
|
stateObj, err := v1.InitializeFromProtoUnsafe(ðpb.BeaconState{
|
|
Validators: []*ethpb.Validator{
|
|
{
|
|
PublicKey: pubKey1,
|
|
ActivationEligibilityEpoch: 1,
|
|
},
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
vs := &Server{
|
|
DepositFetcher: depositCache,
|
|
BlockFetcher: p,
|
|
HeadFetcher: &mockChain.ChainService{
|
|
State: stateObj,
|
|
},
|
|
Eth1InfoFetcher: p,
|
|
}
|
|
req := ðpb.ValidatorStatusRequest{
|
|
PublicKey: pubKey1,
|
|
}
|
|
resp, err := vs.ValidatorStatus(context.Background(), req)
|
|
require.NoError(t, err, "Could not get validator status")
|
|
assert.Equal(t, ethpb.ValidatorStatus_PARTIALLY_DEPOSITED, resp.Status)
|
|
}
|
|
|
|
func TestValidatorStatus_Pending(t *testing.T) {
|
|
ctx := context.Background()
|
|
|
|
pubKey := pubKey(1)
|
|
block := util.NewBeaconBlock()
|
|
genesisRoot, err := block.Block.HashTreeRoot()
|
|
require.NoError(t, err, "Could not get signing root")
|
|
// Pending active because activation epoch is still defaulted at far future slot.
|
|
state, err := util.NewBeaconState()
|
|
require.NoError(t, err)
|
|
require.NoError(t, state.SetSlot(5000))
|
|
err = state.SetValidators([]*ethpb.Validator{
|
|
{
|
|
ActivationEpoch: params.BeaconConfig().FarFutureEpoch,
|
|
ExitEpoch: params.BeaconConfig().FarFutureEpoch,
|
|
WithdrawableEpoch: params.BeaconConfig().FarFutureEpoch,
|
|
PublicKey: pubKey,
|
|
WithdrawalCredentials: make([]byte, 32),
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
|
|
depData := ðpb.Deposit_Data{
|
|
PublicKey: pubKey,
|
|
Signature: bytesutil.PadTo([]byte("hi"), 96),
|
|
WithdrawalCredentials: bytesutil.PadTo([]byte("hey"), 32),
|
|
}
|
|
|
|
deposit := ðpb.Deposit{
|
|
Data: depData,
|
|
}
|
|
depositTrie, err := trie.NewTrie(params.BeaconConfig().DepositContractTreeDepth)
|
|
require.NoError(t, err, "Could not setup deposit trie")
|
|
depositCache, err := depositcache.New()
|
|
require.NoError(t, err)
|
|
|
|
assert.NoError(t, depositCache.InsertDeposit(ctx, deposit, 0 /*blockNum*/, 0, depositTrie.HashTreeRoot()))
|
|
|
|
height := time.Unix(int64(params.BeaconConfig().Eth1FollowDistance), 0).Unix()
|
|
p := &mockPOW.POWChain{
|
|
TimesByHeight: map[int]uint64{
|
|
0: uint64(height),
|
|
},
|
|
}
|
|
vs := &Server{
|
|
ChainStartFetcher: p,
|
|
BlockFetcher: p,
|
|
Eth1InfoFetcher: p,
|
|
DepositFetcher: depositCache,
|
|
HeadFetcher: &mockChain.ChainService{State: state, Root: genesisRoot[:]},
|
|
}
|
|
req := ðpb.ValidatorStatusRequest{
|
|
PublicKey: pubKey,
|
|
}
|
|
resp, err := vs.ValidatorStatus(context.Background(), req)
|
|
require.NoError(t, err, "Could not get validator status")
|
|
assert.Equal(t, ethpb.ValidatorStatus_PENDING, resp.Status)
|
|
}
|
|
|
|
func TestValidatorStatus_Active(t *testing.T) {
|
|
// This test breaks if it doesnt use mainnet config
|
|
params.SetupTestConfigCleanup(t)
|
|
params.OverrideBeaconConfig(params.MainnetConfig())
|
|
ctx := context.Background()
|
|
|
|
pubKey := pubKey(1)
|
|
|
|
depData := ðpb.Deposit_Data{
|
|
PublicKey: pubKey,
|
|
Signature: bytesutil.PadTo([]byte("hi"), 96),
|
|
WithdrawalCredentials: bytesutil.PadTo([]byte("hey"), 32),
|
|
}
|
|
|
|
deposit := ðpb.Deposit{
|
|
Data: depData,
|
|
}
|
|
depositTrie, err := trie.NewTrie(params.BeaconConfig().DepositContractTreeDepth)
|
|
require.NoError(t, err, "Could not setup deposit trie")
|
|
depositCache, err := depositcache.New()
|
|
require.NoError(t, err)
|
|
|
|
assert.NoError(t, depositCache.InsertDeposit(ctx, deposit, 0 /*blockNum*/, 0, depositTrie.HashTreeRoot()))
|
|
|
|
// Active because activation epoch <= current epoch < exit epoch.
|
|
activeEpoch := helpers.ActivationExitEpoch(0)
|
|
|
|
block := util.NewBeaconBlock()
|
|
genesisRoot, err := block.Block.HashTreeRoot()
|
|
require.NoError(t, err, "Could not get signing root")
|
|
|
|
state := ðpb.BeaconState{
|
|
GenesisTime: uint64(time.Unix(0, 0).Unix()),
|
|
Slot: 10000,
|
|
Validators: []*ethpb.Validator{{
|
|
ActivationEpoch: activeEpoch,
|
|
ExitEpoch: params.BeaconConfig().FarFutureEpoch,
|
|
WithdrawableEpoch: params.BeaconConfig().FarFutureEpoch,
|
|
PublicKey: pubKey},
|
|
}}
|
|
stateObj, err := v1.InitializeFromProtoUnsafe(state)
|
|
require.NoError(t, err)
|
|
|
|
timestamp := time.Unix(int64(params.BeaconConfig().Eth1FollowDistance), 0).Unix()
|
|
p := &mockPOW.POWChain{
|
|
TimesByHeight: map[int]uint64{
|
|
int(params.BeaconConfig().Eth1FollowDistance): uint64(timestamp),
|
|
},
|
|
}
|
|
vs := &Server{
|
|
ChainStartFetcher: p,
|
|
BlockFetcher: p,
|
|
Eth1InfoFetcher: p,
|
|
DepositFetcher: depositCache,
|
|
HeadFetcher: &mockChain.ChainService{State: stateObj, Root: genesisRoot[:]},
|
|
}
|
|
req := ðpb.ValidatorStatusRequest{
|
|
PublicKey: pubKey,
|
|
}
|
|
resp, err := vs.ValidatorStatus(context.Background(), req)
|
|
require.NoError(t, err, "Could not get validator status")
|
|
|
|
expected := ðpb.ValidatorStatusResponse{
|
|
Status: ethpb.ValidatorStatus_ACTIVE,
|
|
ActivationEpoch: 5,
|
|
}
|
|
if !proto.Equal(resp, expected) {
|
|
t.Errorf("Wanted %v, got %v", expected, resp)
|
|
}
|
|
}
|
|
|
|
func TestValidatorStatus_Exiting(t *testing.T) {
|
|
ctx := context.Background()
|
|
|
|
pubKey := pubKey(1)
|
|
|
|
// Initiated exit because validator exit epoch and withdrawable epoch are not FAR_FUTURE_EPOCH
|
|
slot := types.Slot(10000)
|
|
epoch := slots.ToEpoch(slot)
|
|
exitEpoch := helpers.ActivationExitEpoch(epoch)
|
|
withdrawableEpoch := exitEpoch + params.BeaconConfig().MinValidatorWithdrawabilityDelay
|
|
block := util.NewBeaconBlock()
|
|
genesisRoot, err := block.Block.HashTreeRoot()
|
|
require.NoError(t, err, "Could not get signing root")
|
|
|
|
state := ðpb.BeaconState{
|
|
Slot: slot,
|
|
Validators: []*ethpb.Validator{{
|
|
PublicKey: pubKey,
|
|
ActivationEpoch: 0,
|
|
ExitEpoch: exitEpoch,
|
|
WithdrawableEpoch: withdrawableEpoch},
|
|
}}
|
|
stateObj, err := v1.InitializeFromProtoUnsafe(state)
|
|
require.NoError(t, err)
|
|
depData := ðpb.Deposit_Data{
|
|
PublicKey: pubKey,
|
|
Signature: bytesutil.PadTo([]byte("hi"), 96),
|
|
WithdrawalCredentials: bytesutil.PadTo([]byte("hey"), 32),
|
|
}
|
|
|
|
deposit := ðpb.Deposit{
|
|
Data: depData,
|
|
}
|
|
depositTrie, err := trie.NewTrie(params.BeaconConfig().DepositContractTreeDepth)
|
|
require.NoError(t, err, "Could not setup deposit trie")
|
|
depositCache, err := depositcache.New()
|
|
require.NoError(t, err)
|
|
|
|
assert.NoError(t, depositCache.InsertDeposit(ctx, deposit, 0 /*blockNum*/, 0, depositTrie.HashTreeRoot()))
|
|
height := time.Unix(int64(params.BeaconConfig().Eth1FollowDistance), 0).Unix()
|
|
p := &mockPOW.POWChain{
|
|
TimesByHeight: map[int]uint64{
|
|
0: uint64(height),
|
|
},
|
|
}
|
|
vs := &Server{
|
|
ChainStartFetcher: p,
|
|
BlockFetcher: p,
|
|
Eth1InfoFetcher: p,
|
|
DepositFetcher: depositCache,
|
|
HeadFetcher: &mockChain.ChainService{State: stateObj, Root: genesisRoot[:]},
|
|
}
|
|
req := ðpb.ValidatorStatusRequest{
|
|
PublicKey: pubKey,
|
|
}
|
|
resp, err := vs.ValidatorStatus(context.Background(), req)
|
|
require.NoError(t, err, "Could not get validator status")
|
|
assert.Equal(t, ethpb.ValidatorStatus_EXITING, resp.Status)
|
|
}
|
|
|
|
func TestValidatorStatus_Slashing(t *testing.T) {
|
|
ctx := context.Background()
|
|
|
|
pubKey := pubKey(1)
|
|
|
|
// Exit slashed because slashed is true, exit epoch is =< current epoch and withdrawable epoch > epoch .
|
|
slot := types.Slot(10000)
|
|
epoch := slots.ToEpoch(slot)
|
|
block := util.NewBeaconBlock()
|
|
genesisRoot, err := block.Block.HashTreeRoot()
|
|
require.NoError(t, err, "Could not get signing root")
|
|
|
|
state := ðpb.BeaconState{
|
|
Slot: slot,
|
|
Validators: []*ethpb.Validator{{
|
|
Slashed: true,
|
|
PublicKey: pubKey,
|
|
WithdrawableEpoch: epoch + 1},
|
|
}}
|
|
stateObj, err := v1.InitializeFromProtoUnsafe(state)
|
|
require.NoError(t, err)
|
|
depData := ðpb.Deposit_Data{
|
|
PublicKey: pubKey,
|
|
Signature: bytesutil.PadTo([]byte("hi"), 96),
|
|
WithdrawalCredentials: bytesutil.PadTo([]byte("hey"), 32),
|
|
}
|
|
|
|
deposit := ðpb.Deposit{
|
|
Data: depData,
|
|
}
|
|
depositTrie, err := trie.NewTrie(params.BeaconConfig().DepositContractTreeDepth)
|
|
require.NoError(t, err, "Could not setup deposit trie")
|
|
depositCache, err := depositcache.New()
|
|
require.NoError(t, err)
|
|
|
|
assert.NoError(t, depositCache.InsertDeposit(ctx, deposit, 0 /*blockNum*/, 0, depositTrie.HashTreeRoot()))
|
|
height := time.Unix(int64(params.BeaconConfig().Eth1FollowDistance), 0).Unix()
|
|
p := &mockPOW.POWChain{
|
|
TimesByHeight: map[int]uint64{
|
|
0: uint64(height),
|
|
},
|
|
}
|
|
vs := &Server{
|
|
ChainStartFetcher: p,
|
|
Eth1InfoFetcher: p,
|
|
DepositFetcher: depositCache,
|
|
BlockFetcher: p,
|
|
HeadFetcher: &mockChain.ChainService{State: stateObj, Root: genesisRoot[:]},
|
|
}
|
|
req := ðpb.ValidatorStatusRequest{
|
|
PublicKey: pubKey,
|
|
}
|
|
resp, err := vs.ValidatorStatus(context.Background(), req)
|
|
require.NoError(t, err, "Could not get validator status")
|
|
assert.Equal(t, ethpb.ValidatorStatus_EXITED, resp.Status)
|
|
}
|
|
|
|
func TestValidatorStatus_Exited(t *testing.T) {
|
|
ctx := context.Background()
|
|
|
|
pubKey := pubKey(1)
|
|
|
|
// Exit because only exit epoch is =< current epoch.
|
|
slot := types.Slot(10000)
|
|
epoch := slots.ToEpoch(slot)
|
|
block := util.NewBeaconBlock()
|
|
genesisRoot, err := block.Block.HashTreeRoot()
|
|
require.NoError(t, err, "Could not get signing root")
|
|
params.SetupTestConfigCleanup(t)
|
|
params.OverrideBeaconConfig(params.MainnetConfig())
|
|
state, err := util.NewBeaconState()
|
|
require.NoError(t, err)
|
|
require.NoError(t, state.SetSlot(slot))
|
|
err = state.SetValidators([]*ethpb.Validator{{
|
|
PublicKey: pubKey,
|
|
WithdrawableEpoch: epoch + 1,
|
|
WithdrawalCredentials: make([]byte, 32)},
|
|
})
|
|
require.NoError(t, err)
|
|
depData := ðpb.Deposit_Data{
|
|
PublicKey: pubKey,
|
|
Signature: bytesutil.PadTo([]byte("hi"), 96),
|
|
WithdrawalCredentials: bytesutil.PadTo([]byte("hey"), 32),
|
|
}
|
|
|
|
deposit := ðpb.Deposit{
|
|
Data: depData,
|
|
}
|
|
depositTrie, err := trie.NewTrie(params.BeaconConfig().DepositContractTreeDepth)
|
|
require.NoError(t, err, "Could not setup deposit trie")
|
|
depositCache, err := depositcache.New()
|
|
require.NoError(t, err)
|
|
|
|
assert.NoError(t, depositCache.InsertDeposit(ctx, deposit, 0 /*blockNum*/, 0, depositTrie.HashTreeRoot()))
|
|
height := time.Unix(int64(params.BeaconConfig().Eth1FollowDistance), 0).Unix()
|
|
p := &mockPOW.POWChain{
|
|
TimesByHeight: map[int]uint64{
|
|
0: uint64(height),
|
|
},
|
|
}
|
|
vs := &Server{
|
|
ChainStartFetcher: p,
|
|
Eth1InfoFetcher: p,
|
|
BlockFetcher: p,
|
|
DepositFetcher: depositCache,
|
|
HeadFetcher: &mockChain.ChainService{State: state, Root: genesisRoot[:]},
|
|
}
|
|
req := ðpb.ValidatorStatusRequest{
|
|
PublicKey: pubKey,
|
|
}
|
|
resp, err := vs.ValidatorStatus(context.Background(), req)
|
|
require.NoError(t, err, "Could not get validator status")
|
|
assert.Equal(t, ethpb.ValidatorStatus_EXITED, resp.Status)
|
|
}
|
|
|
|
func TestValidatorStatus_UnknownStatus(t *testing.T) {
|
|
pubKey := pubKey(1)
|
|
depositCache, err := depositcache.New()
|
|
require.NoError(t, err)
|
|
|
|
stateObj, err := v1.InitializeFromProtoUnsafe(ðpb.BeaconState{
|
|
Slot: 0,
|
|
})
|
|
require.NoError(t, err)
|
|
vs := &Server{
|
|
DepositFetcher: depositCache,
|
|
Eth1InfoFetcher: &mockPOW.POWChain{},
|
|
HeadFetcher: &mockChain.ChainService{
|
|
State: stateObj,
|
|
},
|
|
}
|
|
req := ðpb.ValidatorStatusRequest{
|
|
PublicKey: pubKey,
|
|
}
|
|
resp, err := vs.ValidatorStatus(context.Background(), req)
|
|
require.NoError(t, err, "Could not get validator status")
|
|
assert.Equal(t, ethpb.ValidatorStatus_UNKNOWN_STATUS, resp.Status)
|
|
}
|
|
|
|
func TestActivationStatus_OK(t *testing.T) {
|
|
ctx := context.Background()
|
|
|
|
deposits, _, err := util.DeterministicDepositsAndKeys(4)
|
|
require.NoError(t, err)
|
|
pubKeys := [][]byte{deposits[0].Data.PublicKey, deposits[1].Data.PublicKey, deposits[2].Data.PublicKey, deposits[3].Data.PublicKey}
|
|
stateObj, err := v1.InitializeFromProtoUnsafe(ðpb.BeaconState{
|
|
Slot: 4000,
|
|
Validators: []*ethpb.Validator{
|
|
{
|
|
ActivationEpoch: 0,
|
|
ExitEpoch: params.BeaconConfig().FarFutureEpoch,
|
|
PublicKey: pubKeys[0],
|
|
},
|
|
{
|
|
ActivationEpoch: 0,
|
|
ExitEpoch: params.BeaconConfig().FarFutureEpoch,
|
|
PublicKey: pubKeys[1],
|
|
},
|
|
{
|
|
ActivationEligibilityEpoch: 700,
|
|
ExitEpoch: params.BeaconConfig().FarFutureEpoch,
|
|
PublicKey: pubKeys[3],
|
|
EffectiveBalance: params.BeaconConfig().MaxEffectiveBalance,
|
|
},
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
block := util.NewBeaconBlock()
|
|
genesisRoot, err := block.Block.HashTreeRoot()
|
|
require.NoError(t, err, "Could not get signing root")
|
|
dep := deposits[0]
|
|
depositTrie, err := trie.NewTrie(params.BeaconConfig().DepositContractTreeDepth)
|
|
require.NoError(t, err, "Could not setup deposit trie")
|
|
depositCache, err := depositcache.New()
|
|
require.NoError(t, err)
|
|
|
|
assert.NoError(t, depositCache.InsertDeposit(ctx, dep, 10 /*blockNum*/, 0, depositTrie.HashTreeRoot()))
|
|
|
|
dep = deposits[2]
|
|
assert.NoError(t, depositTrie.Insert(dep.Data.Signature, 15))
|
|
assert.NoError(t, depositCache.InsertDeposit(context.Background(), dep, 0, 1, depositTrie.HashTreeRoot()))
|
|
|
|
vs := &Server{
|
|
Ctx: context.Background(),
|
|
ChainStartFetcher: &mockPOW.POWChain{},
|
|
BlockFetcher: &mockPOW.POWChain{},
|
|
Eth1InfoFetcher: &mockPOW.POWChain{},
|
|
DepositFetcher: depositCache,
|
|
HeadFetcher: &mockChain.ChainService{State: stateObj, Root: genesisRoot[:]},
|
|
}
|
|
activeExists, response, err := vs.activationStatus(context.Background(), pubKeys)
|
|
require.NoError(t, err)
|
|
require.Equal(t, true, activeExists, "No activated validator exists when there was supposed to be 2")
|
|
if response[0].Status.Status != ethpb.ValidatorStatus_ACTIVE {
|
|
t.Errorf("Validator with pubkey %#x is not activated and instead has this status: %s",
|
|
response[0].PublicKey, response[0].Status.Status.String())
|
|
}
|
|
if response[0].Index != 0 {
|
|
t.Errorf("Validator with pubkey %#x is expected to have index %d, received %d", response[0].PublicKey, 0, response[0].Index)
|
|
}
|
|
|
|
if response[1].Status.Status != ethpb.ValidatorStatus_ACTIVE {
|
|
t.Errorf("Validator with pubkey %#x was activated when not supposed to",
|
|
response[1].PublicKey)
|
|
}
|
|
if response[1].Index != 1 {
|
|
t.Errorf("Validator with pubkey %#x is expected to have index %d, received %d", response[1].PublicKey, 1, response[1].Index)
|
|
}
|
|
|
|
if response[2].Status.Status != ethpb.ValidatorStatus_DEPOSITED {
|
|
t.Errorf("Validator with pubkey %#x is not unknown and instead has this status: %s",
|
|
response[2].PublicKey, response[2].Status.Status.String())
|
|
}
|
|
if uint64(response[2].Index) != uint64(params.BeaconConfig().FarFutureEpoch) {
|
|
t.Errorf("Validator with pubkey %#x is expected to have index %d, received %d", response[2].PublicKey, params.BeaconConfig().FarFutureEpoch, response[2].Index)
|
|
}
|
|
|
|
if response[3].Status.Status != ethpb.ValidatorStatus_DEPOSITED {
|
|
t.Errorf("Validator with pubkey %#x was not deposited and has this status: %s",
|
|
response[3].PublicKey, response[3].Status.Status.String())
|
|
}
|
|
if response[3].Index != 2 {
|
|
t.Errorf("Validator with pubkey %#x is expected to have index %d, received %d", response[3].PublicKey, 2, response[3].Index)
|
|
}
|
|
}
|
|
|
|
func TestValidatorStatus_CorrectActivationQueue(t *testing.T) {
|
|
ctx := context.Background()
|
|
|
|
pbKey := pubKey(5)
|
|
block := util.NewBeaconBlock()
|
|
genesisRoot, err := block.Block.HashTreeRoot()
|
|
require.NoError(t, err, "Could not get signing root")
|
|
currentSlot := types.Slot(5000)
|
|
// Pending active because activation epoch is still defaulted at far future slot.
|
|
validators := []*ethpb.Validator{
|
|
{
|
|
ActivationEpoch: 0,
|
|
PublicKey: pubKey(0),
|
|
WithdrawalCredentials: make([]byte, 32),
|
|
ExitEpoch: params.BeaconConfig().FarFutureEpoch,
|
|
WithdrawableEpoch: params.BeaconConfig().FarFutureEpoch,
|
|
},
|
|
{
|
|
ActivationEpoch: 0,
|
|
PublicKey: pubKey(1),
|
|
WithdrawalCredentials: make([]byte, 32),
|
|
ExitEpoch: params.BeaconConfig().FarFutureEpoch,
|
|
WithdrawableEpoch: params.BeaconConfig().FarFutureEpoch,
|
|
},
|
|
{
|
|
ActivationEpoch: 0,
|
|
PublicKey: pubKey(2),
|
|
WithdrawalCredentials: make([]byte, 32),
|
|
ExitEpoch: params.BeaconConfig().FarFutureEpoch,
|
|
WithdrawableEpoch: params.BeaconConfig().FarFutureEpoch,
|
|
},
|
|
{
|
|
ActivationEpoch: 0,
|
|
PublicKey: pubKey(3),
|
|
WithdrawalCredentials: make([]byte, 32),
|
|
ExitEpoch: params.BeaconConfig().FarFutureEpoch,
|
|
WithdrawableEpoch: params.BeaconConfig().FarFutureEpoch,
|
|
},
|
|
{
|
|
ActivationEpoch: types.Epoch(currentSlot/params.BeaconConfig().SlotsPerEpoch + 1),
|
|
PublicKey: pbKey,
|
|
WithdrawalCredentials: make([]byte, 32),
|
|
ExitEpoch: params.BeaconConfig().FarFutureEpoch,
|
|
WithdrawableEpoch: params.BeaconConfig().FarFutureEpoch,
|
|
},
|
|
{
|
|
ActivationEpoch: types.Epoch(currentSlot/params.BeaconConfig().SlotsPerEpoch + 4),
|
|
PublicKey: pubKey(5),
|
|
WithdrawalCredentials: make([]byte, 32),
|
|
ExitEpoch: params.BeaconConfig().FarFutureEpoch,
|
|
WithdrawableEpoch: params.BeaconConfig().FarFutureEpoch,
|
|
},
|
|
}
|
|
state, err := util.NewBeaconState()
|
|
require.NoError(t, err)
|
|
require.NoError(t, state.SetValidators(validators))
|
|
require.NoError(t, state.SetSlot(currentSlot))
|
|
|
|
depositTrie, err := trie.NewTrie(params.BeaconConfig().DepositContractTreeDepth)
|
|
require.NoError(t, err, "Could not setup deposit trie")
|
|
depositCache, err := depositcache.New()
|
|
require.NoError(t, err)
|
|
|
|
for i := 0; i < 6; i++ {
|
|
depData := ðpb.Deposit_Data{
|
|
PublicKey: pubKey(uint64(i)),
|
|
Signature: bytesutil.PadTo([]byte("hi"), 96),
|
|
WithdrawalCredentials: bytesutil.PadTo([]byte("hey"), 32),
|
|
}
|
|
|
|
deposit := ðpb.Deposit{
|
|
Data: depData,
|
|
}
|
|
assert.NoError(t, depositCache.InsertDeposit(ctx, deposit, 0 /*blockNum*/, int64(i), depositTrie.HashTreeRoot()))
|
|
|
|
}
|
|
|
|
height := time.Unix(int64(params.BeaconConfig().Eth1FollowDistance), 0).Unix()
|
|
p := &mockPOW.POWChain{
|
|
TimesByHeight: map[int]uint64{
|
|
0: uint64(height),
|
|
},
|
|
}
|
|
vs := &Server{
|
|
ChainStartFetcher: p,
|
|
BlockFetcher: p,
|
|
Eth1InfoFetcher: p,
|
|
DepositFetcher: depositCache,
|
|
HeadFetcher: &mockChain.ChainService{State: state, Root: genesisRoot[:]},
|
|
}
|
|
req := ðpb.ValidatorStatusRequest{
|
|
PublicKey: pbKey,
|
|
}
|
|
resp, err := vs.ValidatorStatus(context.Background(), req)
|
|
require.NoError(t, err, "Could not get validator status")
|
|
assert.Equal(t, ethpb.ValidatorStatus_PENDING, resp.Status)
|
|
assert.Equal(t, uint64(2), resp.PositionInActivationQueue, "Unexpected position in activation queue")
|
|
}
|
|
|
|
func TestMultipleValidatorStatus_Pubkeys(t *testing.T) {
|
|
ctx := context.Background()
|
|
|
|
deposits, _, err := util.DeterministicDepositsAndKeys(6)
|
|
require.NoError(t, err)
|
|
pubKeys := [][]byte{
|
|
deposits[0].Data.PublicKey,
|
|
deposits[1].Data.PublicKey,
|
|
deposits[2].Data.PublicKey,
|
|
deposits[3].Data.PublicKey,
|
|
deposits[4].Data.PublicKey,
|
|
deposits[5].Data.PublicKey,
|
|
}
|
|
stateObj, err := v1.InitializeFromProtoUnsafe(ðpb.BeaconState{
|
|
Slot: 4000,
|
|
Validators: []*ethpb.Validator{
|
|
{
|
|
ActivationEpoch: 0,
|
|
ExitEpoch: params.BeaconConfig().FarFutureEpoch,
|
|
PublicKey: pubKeys[0],
|
|
},
|
|
{
|
|
ActivationEpoch: 0,
|
|
ExitEpoch: params.BeaconConfig().FarFutureEpoch,
|
|
PublicKey: pubKeys[1],
|
|
},
|
|
{
|
|
ActivationEligibilityEpoch: 700,
|
|
ExitEpoch: params.BeaconConfig().FarFutureEpoch,
|
|
PublicKey: pubKeys[3],
|
|
EffectiveBalance: params.BeaconConfig().MaxEffectiveBalance,
|
|
},
|
|
{
|
|
ActivationEligibilityEpoch: 700,
|
|
ExitEpoch: params.BeaconConfig().FarFutureEpoch,
|
|
PublicKey: pubKeys[4],
|
|
EffectiveBalance: params.BeaconConfig().MinDepositAmount,
|
|
},
|
|
{
|
|
ActivationEligibilityEpoch: 700,
|
|
ExitEpoch: params.BeaconConfig().FarFutureEpoch,
|
|
PublicKey: pubKeys[5],
|
|
},
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
block := util.NewBeaconBlock()
|
|
genesisRoot, err := block.Block.HashTreeRoot()
|
|
require.NoError(t, err, "Could not get signing root")
|
|
depositTrie, err := trie.NewTrie(params.BeaconConfig().DepositContractTreeDepth)
|
|
require.NoError(t, err, "Could not setup deposit trie")
|
|
depositCache, err := depositcache.New()
|
|
require.NoError(t, err)
|
|
|
|
dep := deposits[0]
|
|
assert.NoError(t, depositCache.InsertDeposit(ctx, dep, 10 /*blockNum*/, 0, depositTrie.HashTreeRoot()))
|
|
dep = deposits[2]
|
|
assert.NoError(t, depositTrie.Insert(dep.Data.Signature, 15))
|
|
assert.NoError(t, depositCache.InsertDeposit(context.Background(), dep, 0, 1, depositTrie.HashTreeRoot()))
|
|
|
|
vs := &Server{
|
|
Ctx: context.Background(),
|
|
ChainStartFetcher: &mockPOW.POWChain{},
|
|
BlockFetcher: &mockPOW.POWChain{},
|
|
Eth1InfoFetcher: &mockPOW.POWChain{},
|
|
DepositFetcher: depositCache,
|
|
HeadFetcher: &mockChain.ChainService{State: stateObj, Root: genesisRoot[:]},
|
|
SyncChecker: &mockSync.Sync{IsSyncing: false},
|
|
}
|
|
|
|
want := []*ethpb.ValidatorStatusResponse{
|
|
{
|
|
Status: ethpb.ValidatorStatus_ACTIVE,
|
|
},
|
|
{
|
|
Status: ethpb.ValidatorStatus_ACTIVE,
|
|
},
|
|
{
|
|
Status: ethpb.ValidatorStatus_DEPOSITED,
|
|
ActivationEpoch: 18446744073709551615,
|
|
},
|
|
{
|
|
Status: ethpb.ValidatorStatus_DEPOSITED,
|
|
},
|
|
{
|
|
Status: ethpb.ValidatorStatus_PARTIALLY_DEPOSITED,
|
|
},
|
|
{
|
|
Status: ethpb.ValidatorStatus_PENDING,
|
|
},
|
|
}
|
|
|
|
req := ðpb.MultipleValidatorStatusRequest{PublicKeys: pubKeys}
|
|
response, err := vs.MultipleValidatorStatus(context.Background(), req)
|
|
require.NoError(t, err)
|
|
|
|
assert.Equal(t, len(response.PublicKeys), len(pubKeys))
|
|
for i, resp := range response.PublicKeys {
|
|
require.DeepEqual(t, pubKeys[i], resp)
|
|
}
|
|
assert.Equal(t, len(pubKeys), len(response.Statuses))
|
|
for i, resp := range response.Statuses {
|
|
if !proto.Equal(want[i], resp) {
|
|
t.Fatalf("Wanted %v\n Recieved: %v\n", want[i], resp)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestMultipleValidatorStatus_Indices(t *testing.T) {
|
|
slot := types.Slot(10000)
|
|
epoch := slots.ToEpoch(slot)
|
|
pubKeys := [][]byte{pubKey(1), pubKey(2), pubKey(3), pubKey(4), pubKey(5), pubKey(6), pubKey(7)}
|
|
beaconState := ðpb.BeaconState{
|
|
Slot: 4000,
|
|
Validators: []*ethpb.Validator{
|
|
{
|
|
ActivationEpoch: 0,
|
|
ExitEpoch: params.BeaconConfig().FarFutureEpoch,
|
|
PublicKey: pubKeys[0],
|
|
},
|
|
{
|
|
ActivationEpoch: 0,
|
|
ExitEpoch: params.BeaconConfig().FarFutureEpoch,
|
|
PublicKey: pubKeys[1],
|
|
},
|
|
{
|
|
ActivationEligibilityEpoch: 700,
|
|
ExitEpoch: params.BeaconConfig().FarFutureEpoch,
|
|
PublicKey: pubKeys[2],
|
|
EffectiveBalance: params.BeaconConfig().MaxEffectiveBalance,
|
|
},
|
|
{
|
|
Slashed: true,
|
|
ExitEpoch: epoch + 1,
|
|
PublicKey: pubKeys[3],
|
|
},
|
|
{
|
|
ActivationEligibilityEpoch: 700,
|
|
ExitEpoch: params.BeaconConfig().FarFutureEpoch,
|
|
PublicKey: pubKeys[4],
|
|
EffectiveBalance: params.BeaconConfig().MinDepositAmount,
|
|
},
|
|
{
|
|
ActivationEligibilityEpoch: 700,
|
|
ExitEpoch: params.BeaconConfig().FarFutureEpoch,
|
|
PublicKey: pubKeys[5],
|
|
},
|
|
},
|
|
}
|
|
stateObj, err := v1.InitializeFromProtoUnsafe(beaconState)
|
|
require.NoError(t, err)
|
|
block := util.NewBeaconBlock()
|
|
genesisRoot, err := block.Block.HashTreeRoot()
|
|
require.NoError(t, err, "Could not get signing root")
|
|
|
|
vs := &Server{
|
|
Ctx: context.Background(),
|
|
ChainStartFetcher: &mockPOW.POWChain{},
|
|
BlockFetcher: &mockPOW.POWChain{},
|
|
Eth1InfoFetcher: &mockPOW.POWChain{},
|
|
HeadFetcher: &mockChain.ChainService{State: stateObj, Root: genesisRoot[:]},
|
|
SyncChecker: &mockSync.Sync{IsSyncing: false},
|
|
}
|
|
|
|
want := []*ethpb.ValidatorStatusResponse{
|
|
{
|
|
Status: ethpb.ValidatorStatus_ACTIVE,
|
|
},
|
|
{
|
|
Status: ethpb.ValidatorStatus_ACTIVE,
|
|
},
|
|
{
|
|
Status: ethpb.ValidatorStatus_DEPOSITED,
|
|
},
|
|
{
|
|
Status: ethpb.ValidatorStatus_SLASHING,
|
|
},
|
|
{
|
|
Status: ethpb.ValidatorStatus_PARTIALLY_DEPOSITED,
|
|
},
|
|
{
|
|
Status: ethpb.ValidatorStatus_PENDING,
|
|
},
|
|
}
|
|
|
|
// Note: Index 6 should be skipped.
|
|
req := ðpb.MultipleValidatorStatusRequest{Indices: []int64{0, 1, 2, 3, 4, 5, 6}}
|
|
response, err := vs.MultipleValidatorStatus(context.Background(), req)
|
|
require.NoError(t, err)
|
|
|
|
assert.Equal(t, len(beaconState.Validators), len(response.PublicKeys))
|
|
for i, resp := range response.PublicKeys {
|
|
expected := beaconState.Validators[i].PublicKey
|
|
require.DeepEqual(t, expected, resp)
|
|
}
|
|
assert.Equal(t, len(beaconState.Validators), len(response.Statuses))
|
|
for i, resp := range response.Statuses {
|
|
if !proto.Equal(want[i], resp) {
|
|
t.Fatalf("Wanted %v\n Recieved: %v\n", want[i], resp)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestValidatorStatus_Invalid(t *testing.T) {
|
|
ctx := context.Background()
|
|
deposits, _, err := util.DeterministicDepositsAndKeys(1)
|
|
require.NoError(t, err, "Could not generate deposits and keys")
|
|
deposit := deposits[0]
|
|
pubKey1 := deposit.Data.PublicKey
|
|
deposit.Data.Signature = deposit.Data.Signature[1:]
|
|
depositTrie, err := trie.NewTrie(params.BeaconConfig().DepositContractTreeDepth)
|
|
require.NoError(t, err, "Could not setup deposit trie")
|
|
depositCache, err := depositcache.New()
|
|
require.NoError(t, err)
|
|
|
|
assert.NoError(t, depositCache.InsertDeposit(ctx, deposit, 0 /*blockNum*/, 0, depositTrie.HashTreeRoot()))
|
|
height := time.Unix(int64(params.BeaconConfig().Eth1FollowDistance), 0).Unix()
|
|
p := &mockPOW.POWChain{
|
|
TimesByHeight: map[int]uint64{
|
|
0: uint64(height),
|
|
},
|
|
}
|
|
stateObj, err := v1.InitializeFromProtoUnsafe(ðpb.BeaconState{})
|
|
require.NoError(t, err)
|
|
vs := &Server{
|
|
DepositFetcher: depositCache,
|
|
BlockFetcher: p,
|
|
HeadFetcher: &mockChain.ChainService{
|
|
State: stateObj,
|
|
},
|
|
Eth1InfoFetcher: p,
|
|
}
|
|
req := ðpb.ValidatorStatusRequest{
|
|
PublicKey: pubKey1,
|
|
}
|
|
resp, err := vs.ValidatorStatus(context.Background(), req)
|
|
require.NoError(t, err, "Could not get validator status")
|
|
assert.Equal(t, ethpb.ValidatorStatus_INVALID, resp.Status)
|
|
}
|
|
|
|
func Test_DepositStatus(t *testing.T) {
|
|
assert.Equal(t, depositStatus(0), ethpb.ValidatorStatus_PENDING)
|
|
assert.Equal(t, depositStatus(params.BeaconConfig().MinDepositAmount), ethpb.ValidatorStatus_PARTIALLY_DEPOSITED)
|
|
assert.Equal(t, depositStatus(params.BeaconConfig().MaxEffectiveBalance), ethpb.ValidatorStatus_DEPOSITED)
|
|
}
|
|
|
|
func TestServer_CheckDoppelGanger(t *testing.T) {
|
|
tests := []struct {
|
|
name string
|
|
wantErr bool
|
|
svSetup func(t *testing.T) (*Server, *ethpb.DoppelGangerRequest, *ethpb.DoppelGangerResponse)
|
|
}{
|
|
{
|
|
name: "normal doppelganger request",
|
|
wantErr: false,
|
|
svSetup: func(t *testing.T) (*Server, *ethpb.DoppelGangerRequest, *ethpb.DoppelGangerResponse) {
|
|
mockGen := mockstategen.NewMockService()
|
|
hs, ps, os, keys := createStateSetup(t, 4, mockGen)
|
|
// Previous Epoch State
|
|
for i := 0; i < 3; i++ {
|
|
bal, err := ps.BalanceAtIndex(types.ValidatorIndex(i))
|
|
assert.NoError(t, err)
|
|
// Add 100 gwei, to mock an inactivity leak
|
|
assert.NoError(t, ps.UpdateBalancesAtIndex(types.ValidatorIndex(i), bal+1000000000))
|
|
}
|
|
// Older Epoch State
|
|
for i := 0; i < 3; i++ {
|
|
bal, err := os.BalanceAtIndex(types.ValidatorIndex(i))
|
|
assert.NoError(t, err)
|
|
// Add 200 gwei, to mock an inactivity leak
|
|
assert.NoError(t, os.UpdateBalancesAtIndex(types.ValidatorIndex(i), bal+2000000000))
|
|
}
|
|
vs := &Server{
|
|
StateGen: mockGen,
|
|
HeadFetcher: &mockChain.ChainService{
|
|
State: hs,
|
|
},
|
|
SyncChecker: &mockSync.Sync{IsSyncing: false},
|
|
}
|
|
request := ðpb.DoppelGangerRequest{
|
|
ValidatorRequests: make([]*ethpb.DoppelGangerRequest_ValidatorRequest, 0),
|
|
}
|
|
response := ðpb.DoppelGangerResponse{Responses: make([]*ethpb.DoppelGangerResponse_ValidatorResponse, 0)}
|
|
for i := 0; i < 3; i++ {
|
|
request.ValidatorRequests = append(request.ValidatorRequests, ðpb.DoppelGangerRequest_ValidatorRequest{
|
|
PublicKey: keys[i].PublicKey().Marshal(),
|
|
Epoch: 1,
|
|
SignedRoot: []byte{'A'},
|
|
})
|
|
response.Responses = append(response.Responses, ðpb.DoppelGangerResponse_ValidatorResponse{
|
|
PublicKey: keys[i].PublicKey().Marshal(),
|
|
DuplicateExists: false,
|
|
})
|
|
}
|
|
return vs, request, response
|
|
},
|
|
},
|
|
{
|
|
name: "doppelganger exists current epoch",
|
|
wantErr: false,
|
|
svSetup: func(t *testing.T) (*Server, *ethpb.DoppelGangerRequest, *ethpb.DoppelGangerResponse) {
|
|
mockGen := mockstategen.NewMockService()
|
|
|
|
hs, ps, os, keys := createStateSetup(t, 4, mockGen)
|
|
// Previous Epoch State
|
|
for i := 0; i < 2; i++ {
|
|
bal, err := ps.BalanceAtIndex(types.ValidatorIndex(i))
|
|
assert.NoError(t, err)
|
|
// Add 100 gwei, to mock an inactivity leak
|
|
assert.NoError(t, ps.UpdateBalancesAtIndex(types.ValidatorIndex(i), bal+1000000000))
|
|
}
|
|
bal, err := ps.BalanceAtIndex(types.ValidatorIndex(2))
|
|
assert.NoError(t, err)
|
|
// Sub 100 gwei, to mock an active validator.
|
|
assert.NoError(t, ps.UpdateBalancesAtIndex(types.ValidatorIndex(2), bal-1000000000))
|
|
|
|
// Older Epoch State
|
|
for i := 0; i < 2; i++ {
|
|
bal, err := os.BalanceAtIndex(types.ValidatorIndex(i))
|
|
assert.NoError(t, err)
|
|
// Add 200 gwei, to mock an inactivity leak
|
|
assert.NoError(t, os.UpdateBalancesAtIndex(types.ValidatorIndex(i), bal+2000000000))
|
|
}
|
|
bal, err = os.BalanceAtIndex(types.ValidatorIndex(2))
|
|
assert.NoError(t, err)
|
|
// Sub 100 gwei, to mock an active validator.
|
|
assert.NoError(t, os.UpdateBalancesAtIndex(types.ValidatorIndex(2), bal-1000000000))
|
|
|
|
vs := &Server{
|
|
StateGen: mockGen,
|
|
HeadFetcher: &mockChain.ChainService{
|
|
State: hs,
|
|
},
|
|
SyncChecker: &mockSync.Sync{IsSyncing: false},
|
|
}
|
|
request := ðpb.DoppelGangerRequest{
|
|
ValidatorRequests: make([]*ethpb.DoppelGangerRequest_ValidatorRequest, 0),
|
|
}
|
|
response := ðpb.DoppelGangerResponse{Responses: make([]*ethpb.DoppelGangerResponse_ValidatorResponse, 0)}
|
|
for i := 0; i < 2; i++ {
|
|
request.ValidatorRequests = append(request.ValidatorRequests, ðpb.DoppelGangerRequest_ValidatorRequest{
|
|
PublicKey: keys[i].PublicKey().Marshal(),
|
|
Epoch: 1,
|
|
SignedRoot: []byte{'A'},
|
|
})
|
|
response.Responses = append(response.Responses, ðpb.DoppelGangerResponse_ValidatorResponse{
|
|
PublicKey: keys[i].PublicKey().Marshal(),
|
|
DuplicateExists: false,
|
|
})
|
|
}
|
|
|
|
// Add in for duplicate validator
|
|
request.ValidatorRequests = append(request.ValidatorRequests, ðpb.DoppelGangerRequest_ValidatorRequest{
|
|
PublicKey: keys[2].PublicKey().Marshal(),
|
|
Epoch: 1,
|
|
SignedRoot: []byte{'A'},
|
|
})
|
|
response.Responses = append(response.Responses, ðpb.DoppelGangerResponse_ValidatorResponse{
|
|
PublicKey: keys[2].PublicKey().Marshal(),
|
|
DuplicateExists: true,
|
|
})
|
|
return vs, request, response
|
|
},
|
|
},
|
|
{
|
|
name: "doppelganger exists previous epoch",
|
|
wantErr: false,
|
|
svSetup: func(t *testing.T) (*Server, *ethpb.DoppelGangerRequest, *ethpb.DoppelGangerResponse) {
|
|
mockGen := mockstategen.NewMockService()
|
|
|
|
hs, ps, os, keys := createStateSetup(t, 4, mockGen)
|
|
// Previous Epoch State
|
|
for i := 0; i < 2; i++ {
|
|
bal, err := ps.BalanceAtIndex(types.ValidatorIndex(i))
|
|
assert.NoError(t, err)
|
|
// Add 100 gwei, to mock an inactivity leak
|
|
assert.NoError(t, ps.UpdateBalancesAtIndex(types.ValidatorIndex(i), bal+1000000000))
|
|
}
|
|
bal, err := ps.BalanceAtIndex(types.ValidatorIndex(2))
|
|
assert.NoError(t, err)
|
|
// Sub 100 gwei, to mock an active validator.
|
|
assert.NoError(t, ps.UpdateBalancesAtIndex(types.ValidatorIndex(2), bal-1000000000))
|
|
|
|
// Older Epoch State
|
|
for i := 0; i < 2; i++ {
|
|
bal, err := os.BalanceAtIndex(types.ValidatorIndex(i))
|
|
assert.NoError(t, err)
|
|
// Add 200 gwei, to mock an inactivity leak
|
|
assert.NoError(t, os.UpdateBalancesAtIndex(types.ValidatorIndex(i), bal+2000000000))
|
|
}
|
|
bal, err = os.BalanceAtIndex(types.ValidatorIndex(2))
|
|
assert.NoError(t, err)
|
|
// Sub 200 gwei, to mock an active validator.
|
|
assert.NoError(t, os.UpdateBalancesAtIndex(types.ValidatorIndex(2), bal-2000000000))
|
|
|
|
vs := &Server{
|
|
StateGen: mockGen,
|
|
HeadFetcher: &mockChain.ChainService{
|
|
State: hs,
|
|
},
|
|
SyncChecker: &mockSync.Sync{IsSyncing: false},
|
|
}
|
|
request := ðpb.DoppelGangerRequest{
|
|
ValidatorRequests: make([]*ethpb.DoppelGangerRequest_ValidatorRequest, 0),
|
|
}
|
|
response := ðpb.DoppelGangerResponse{Responses: make([]*ethpb.DoppelGangerResponse_ValidatorResponse, 0)}
|
|
for i := 0; i < 2; i++ {
|
|
request.ValidatorRequests = append(request.ValidatorRequests, ðpb.DoppelGangerRequest_ValidatorRequest{
|
|
PublicKey: keys[i].PublicKey().Marshal(),
|
|
Epoch: 1,
|
|
SignedRoot: []byte{'A'},
|
|
})
|
|
response.Responses = append(response.Responses, ðpb.DoppelGangerResponse_ValidatorResponse{
|
|
PublicKey: keys[i].PublicKey().Marshal(),
|
|
DuplicateExists: false,
|
|
})
|
|
}
|
|
|
|
// Add in for duplicate validator
|
|
request.ValidatorRequests = append(request.ValidatorRequests, ðpb.DoppelGangerRequest_ValidatorRequest{
|
|
PublicKey: keys[2].PublicKey().Marshal(),
|
|
Epoch: 1,
|
|
SignedRoot: []byte{'A'},
|
|
})
|
|
response.Responses = append(response.Responses, ðpb.DoppelGangerResponse_ValidatorResponse{
|
|
PublicKey: keys[2].PublicKey().Marshal(),
|
|
DuplicateExists: true,
|
|
})
|
|
return vs, request, response
|
|
},
|
|
},
|
|
{
|
|
name: "multiple doppelganger exists",
|
|
wantErr: false,
|
|
svSetup: func(t *testing.T) (*Server, *ethpb.DoppelGangerRequest, *ethpb.DoppelGangerResponse) {
|
|
mockGen := mockstategen.NewMockService()
|
|
|
|
hs, ps, os, keys := createStateSetup(t, 4, mockGen)
|
|
// Previous Epoch State
|
|
for i := 10; i < 15; i++ {
|
|
bal, err := ps.BalanceAtIndex(types.ValidatorIndex(i))
|
|
assert.NoError(t, err)
|
|
// Add 100 gwei, to mock an inactivity leak
|
|
assert.NoError(t, ps.UpdateBalancesAtIndex(types.ValidatorIndex(i), bal-1000000000))
|
|
}
|
|
|
|
// Older Epoch State
|
|
for i := 10; i < 15; i++ {
|
|
bal, err := os.BalanceAtIndex(types.ValidatorIndex(i))
|
|
assert.NoError(t, err)
|
|
// Add 200 gwei, to mock an inactivity leak
|
|
assert.NoError(t, os.UpdateBalancesAtIndex(types.ValidatorIndex(i), bal-2000000000))
|
|
}
|
|
|
|
vs := &Server{
|
|
StateGen: mockGen,
|
|
HeadFetcher: &mockChain.ChainService{
|
|
State: hs,
|
|
},
|
|
SyncChecker: &mockSync.Sync{IsSyncing: false},
|
|
}
|
|
request := ðpb.DoppelGangerRequest{
|
|
ValidatorRequests: make([]*ethpb.DoppelGangerRequest_ValidatorRequest, 0),
|
|
}
|
|
response := ðpb.DoppelGangerResponse{Responses: make([]*ethpb.DoppelGangerResponse_ValidatorResponse, 0)}
|
|
for i := 10; i < 15; i++ {
|
|
// Add in for duplicate validator
|
|
request.ValidatorRequests = append(request.ValidatorRequests, ðpb.DoppelGangerRequest_ValidatorRequest{
|
|
PublicKey: keys[i].PublicKey().Marshal(),
|
|
Epoch: 1,
|
|
SignedRoot: []byte{'A'},
|
|
})
|
|
response.Responses = append(response.Responses, ðpb.DoppelGangerResponse_ValidatorResponse{
|
|
PublicKey: keys[i].PublicKey().Marshal(),
|
|
DuplicateExists: true,
|
|
})
|
|
}
|
|
|
|
return vs, request, response
|
|
},
|
|
},
|
|
{
|
|
name: "attesters are too recent",
|
|
wantErr: false,
|
|
svSetup: func(t *testing.T) (*Server, *ethpb.DoppelGangerRequest, *ethpb.DoppelGangerResponse) {
|
|
mockGen := mockstategen.NewMockService()
|
|
|
|
hs, _, _, keys := createStateSetup(t, 4, mockGen)
|
|
|
|
vs := &Server{
|
|
StateGen: nil,
|
|
HeadFetcher: &mockChain.ChainService{
|
|
State: hs,
|
|
},
|
|
SyncChecker: &mockSync.Sync{IsSyncing: false},
|
|
}
|
|
request := ðpb.DoppelGangerRequest{
|
|
ValidatorRequests: make([]*ethpb.DoppelGangerRequest_ValidatorRequest, 0),
|
|
}
|
|
response := ðpb.DoppelGangerResponse{Responses: make([]*ethpb.DoppelGangerResponse_ValidatorResponse, 0)}
|
|
for i := 0; i < 15; i++ {
|
|
request.ValidatorRequests = append(request.ValidatorRequests, ðpb.DoppelGangerRequest_ValidatorRequest{
|
|
PublicKey: keys[i].PublicKey().Marshal(),
|
|
Epoch: 2,
|
|
SignedRoot: []byte{'A'},
|
|
})
|
|
response.Responses = append(response.Responses, ðpb.DoppelGangerResponse_ValidatorResponse{
|
|
PublicKey: keys[i].PublicKey().Marshal(),
|
|
DuplicateExists: false,
|
|
})
|
|
}
|
|
|
|
return vs, request, response
|
|
},
|
|
},
|
|
}
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
vs, req, resp := tt.svSetup(t)
|
|
got, err := vs.CheckDoppelGanger(context.Background(), req)
|
|
if (err != nil) != tt.wantErr {
|
|
t.Errorf("CheckDoppelGanger() error = %v, wantErr %v", err, tt.wantErr)
|
|
return
|
|
}
|
|
if !reflect.DeepEqual(got, resp) {
|
|
diff, _ := messagediff.PrettyDiff(resp, got)
|
|
t.Errorf("CheckDoppelGanger() difference = %v", diff)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func createStateSetup(t *testing.T, head types.Epoch, mockgen *mockstategen.MockStateManager) (state.BeaconState,
|
|
state.BeaconState, state.BeaconState, []bls.SecretKey) {
|
|
gs, keys := util.DeterministicGenesisState(t, 64)
|
|
hs := gs.Copy()
|
|
// Head State
|
|
headEpoch := head
|
|
headSlot := types.Slot(headEpoch) * params.BeaconConfig().SlotsPerEpoch
|
|
assert.NoError(t, hs.SetSlot(headSlot))
|
|
assingments, _, err := helpers.CommitteeAssignments(context.Background(), hs, headEpoch)
|
|
assert.NoError(t, err)
|
|
for _, ctr := range assingments {
|
|
pendingAtt := ðpb.PendingAttestation{
|
|
AggregationBits: bitfield.NewBitlist64(uint64(len(ctr.Committee))).ToBitlist().Not(),
|
|
Data: ðpb.AttestationData{
|
|
Slot: ctr.AttesterSlot,
|
|
CommitteeIndex: ctr.CommitteeIndex,
|
|
BeaconBlockRoot: make([]byte, fieldparams.RootLength),
|
|
Source: ðpb.Checkpoint{
|
|
Epoch: 0,
|
|
Root: make([]byte, fieldparams.RootLength),
|
|
},
|
|
Target: ðpb.Checkpoint{
|
|
Epoch: 1,
|
|
Root: make([]byte, fieldparams.RootLength),
|
|
},
|
|
},
|
|
InclusionDelay: 1,
|
|
ProposerIndex: 10,
|
|
}
|
|
assert.NoError(t, hs.AppendCurrentEpochAttestations(pendingAtt))
|
|
|
|
}
|
|
mockgen.StatesBySlot[headSlot] = hs
|
|
|
|
// Previous Epoch State
|
|
prevEpoch := headEpoch - 1
|
|
ps := gs.Copy()
|
|
prevSlot, err := slots.EpochEnd(prevEpoch)
|
|
assert.NoError(t, err)
|
|
assert.NoError(t, ps.SetSlot(prevSlot))
|
|
assingments, _, err = helpers.CommitteeAssignments(context.Background(), ps, prevEpoch)
|
|
assert.NoError(t, err)
|
|
for _, ctr := range assingments {
|
|
pendingAtt := ðpb.PendingAttestation{
|
|
AggregationBits: bitfield.NewBitlist64(uint64(len(ctr.Committee))).ToBitlist().Not(),
|
|
Data: ðpb.AttestationData{
|
|
Slot: ctr.AttesterSlot,
|
|
CommitteeIndex: ctr.CommitteeIndex,
|
|
BeaconBlockRoot: make([]byte, fieldparams.RootLength),
|
|
Source: ðpb.Checkpoint{
|
|
Epoch: 0,
|
|
Root: make([]byte, fieldparams.RootLength),
|
|
},
|
|
Target: ðpb.Checkpoint{
|
|
Epoch: 1,
|
|
Root: make([]byte, fieldparams.RootLength),
|
|
},
|
|
},
|
|
InclusionDelay: 1,
|
|
ProposerIndex: 10,
|
|
}
|
|
assert.NoError(t, ps.AppendCurrentEpochAttestations(pendingAtt))
|
|
|
|
}
|
|
mockgen.StatesBySlot[prevSlot] = ps
|
|
|
|
// Older Epoch State
|
|
olderEpoch := prevEpoch - 1
|
|
os := gs.Copy()
|
|
olderSlot, err := slots.EpochEnd(olderEpoch)
|
|
assert.NoError(t, err)
|
|
assert.NoError(t, os.SetSlot(olderSlot))
|
|
assingments, _, err = helpers.CommitteeAssignments(context.Background(), os, olderEpoch)
|
|
assert.NoError(t, err)
|
|
for _, ctr := range assingments {
|
|
attSlot := ctr.AttesterSlot
|
|
if attSlot == olderSlot {
|
|
continue
|
|
}
|
|
pendingAtt := ðpb.PendingAttestation{
|
|
AggregationBits: bitfield.NewBitlist64(uint64(len(ctr.Committee))).ToBitlist().Not(),
|
|
Data: ðpb.AttestationData{
|
|
Slot: attSlot,
|
|
CommitteeIndex: ctr.CommitteeIndex,
|
|
BeaconBlockRoot: make([]byte, fieldparams.RootLength),
|
|
Source: ðpb.Checkpoint{
|
|
Epoch: 0,
|
|
Root: make([]byte, fieldparams.RootLength),
|
|
},
|
|
Target: ðpb.Checkpoint{
|
|
Epoch: 1,
|
|
Root: make([]byte, fieldparams.RootLength),
|
|
},
|
|
},
|
|
InclusionDelay: 1,
|
|
ProposerIndex: 10,
|
|
}
|
|
assert.NoError(t, os.AppendCurrentEpochAttestations(pendingAtt))
|
|
|
|
}
|
|
mockgen.StatesBySlot[olderSlot] = os
|
|
return hs, ps, os, keys
|
|
}
|