mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-20 16:41:11 +00:00
73443208a1
* Remove native state flag and use native state in spectests * remove feature from tests * use e2e config in slasher simulator * use params.BeaconConfig in testutil * use correct function * use minimal config in go_test * fix TestListValidators * parameterize sync committee bits and aggregation bits * Fix TestServer_ListIndexedAttestations_GenesisEpoch (cherry picked from commit 254ab623dde08ae8886b152facdbbd8889ed79db) * fix more tests * fix even more * moreeee * aaaand more * one more fix * one more * simplify TestGetAltairDuties_UnknownPubkey * comment out problematic test * one more fix * one more * aaaand one more * another * use fieldparams in HydrateBlindedBeaconBlockBodyBellatrix * create new package for mainnet tests * TestServer_GetBellatrixBeaconBlock * change slashed validator index * clear cache in reward_test.go * deprecate flag * create bazel mainnet target * move attester mainnet test to mainnet target * "fix" proposer tests * use minimal config in TestServer_circuitBreakBuilder * fix TestProposer_ProposeBlock_OK * more fixes in validator package * more fixes * more fixes * test code * move TestProposer_GetBeaconBlock_BellatrixEpoch to minimal * finally * remove proposer_bellatrix_mainnet_test.go * fix TestServer_GetBellatrixBeaconBlock_HappyCase * fix TestServer_GetBellatrixBeaconBlock_BuilderCase * Preston needs to fix this! * Revert "Preston needs to fix this!" This reverts commit b03d97a16e3080e254c7b19d7f193d3c600ca869. * remove proto state tests * fix migration tests * static analysis fix * review * remove proto state * swap state in tests * fix BUILD file in /proto/testing * remove metrics test with nil state
228 lines
6.3 KiB
Go
228 lines
6.3 KiB
Go
package blockchain
|
|
|
|
import (
|
|
"context"
|
|
"encoding/binary"
|
|
"errors"
|
|
"testing"
|
|
|
|
"github.com/prysmaticlabs/prysm/v3/beacon-chain/state"
|
|
state_native "github.com/prysmaticlabs/prysm/v3/beacon-chain/state/state-native"
|
|
types "github.com/prysmaticlabs/prysm/v3/consensus-types/primitives"
|
|
"github.com/prysmaticlabs/prysm/v3/encoding/bytesutil"
|
|
ethpb "github.com/prysmaticlabs/prysm/v3/proto/prysm/v1alpha1"
|
|
"github.com/prysmaticlabs/prysm/v3/testing/require"
|
|
"github.com/prysmaticlabs/prysm/v3/time/slots"
|
|
)
|
|
|
|
type testStateOpt func(*ethpb.BeaconStateAltair)
|
|
|
|
func testStateWithValidators(v []*ethpb.Validator) testStateOpt {
|
|
return func(a *ethpb.BeaconStateAltair) {
|
|
a.Validators = v
|
|
}
|
|
}
|
|
|
|
func testStateWithSlot(slot types.Slot) testStateOpt {
|
|
return func(a *ethpb.BeaconStateAltair) {
|
|
a.Slot = slot
|
|
}
|
|
}
|
|
|
|
func testStateFixture(opts ...testStateOpt) state.BeaconState {
|
|
a := ðpb.BeaconStateAltair{}
|
|
for _, o := range opts {
|
|
o(a)
|
|
}
|
|
s, _ := state_native.InitializeFromProtoUnsafeAltair(a)
|
|
return s
|
|
}
|
|
|
|
func generateTestValidators(count int, opts ...func(*ethpb.Validator)) []*ethpb.Validator {
|
|
vs := make([]*ethpb.Validator, count)
|
|
var i uint32 = 0
|
|
for ; i < uint32(count); i++ {
|
|
pk := make([]byte, 48)
|
|
binary.LittleEndian.PutUint32(pk, i)
|
|
v := ðpb.Validator{PublicKey: pk}
|
|
for _, o := range opts {
|
|
o(v)
|
|
}
|
|
vs[i] = v
|
|
}
|
|
return vs
|
|
}
|
|
|
|
func oddValidatorsExpired(currentSlot types.Slot) func(*ethpb.Validator) {
|
|
return func(v *ethpb.Validator) {
|
|
pki := binary.LittleEndian.Uint64(v.PublicKey)
|
|
if pki%2 == 0 {
|
|
v.ExitEpoch = types.Epoch(int(slots.ToEpoch(currentSlot)) + 1)
|
|
} else {
|
|
v.ExitEpoch = types.Epoch(int(slots.ToEpoch(currentSlot)) - 1)
|
|
}
|
|
}
|
|
}
|
|
|
|
func oddValidatorsQueued(currentSlot types.Slot) func(*ethpb.Validator) {
|
|
return func(v *ethpb.Validator) {
|
|
v.ExitEpoch = types.Epoch(int(slots.ToEpoch(currentSlot)) + 1)
|
|
pki := binary.LittleEndian.Uint64(v.PublicKey)
|
|
if pki%2 == 0 {
|
|
v.ActivationEpoch = types.Epoch(int(slots.ToEpoch(currentSlot)) - 1)
|
|
} else {
|
|
v.ActivationEpoch = types.Epoch(int(slots.ToEpoch(currentSlot)) + 1)
|
|
}
|
|
}
|
|
}
|
|
|
|
func allValidatorsValid(currentSlot types.Slot) func(*ethpb.Validator) {
|
|
return func(v *ethpb.Validator) {
|
|
v.ActivationEpoch = types.Epoch(int(slots.ToEpoch(currentSlot)) - 1)
|
|
v.ExitEpoch = types.Epoch(int(slots.ToEpoch(currentSlot)) + 1)
|
|
}
|
|
}
|
|
|
|
func balanceIsKeyTimes2(v *ethpb.Validator) {
|
|
pki := binary.LittleEndian.Uint64(v.PublicKey)
|
|
v.EffectiveBalance = uint64(pki) * 2
|
|
}
|
|
|
|
func testHalfExpiredValidators() ([]*ethpb.Validator, []uint64) {
|
|
balances := []uint64{0, 0, 4, 0, 8, 0, 12, 0, 16, 0}
|
|
return generateTestValidators(10,
|
|
oddValidatorsExpired(types.Slot(99)),
|
|
balanceIsKeyTimes2), balances
|
|
}
|
|
|
|
func testHalfQueuedValidators() ([]*ethpb.Validator, []uint64) {
|
|
balances := []uint64{0, 0, 4, 0, 8, 0, 12, 0, 16, 0}
|
|
return generateTestValidators(10,
|
|
oddValidatorsQueued(types.Slot(99)),
|
|
balanceIsKeyTimes2), balances
|
|
}
|
|
|
|
func testAllValidValidators() ([]*ethpb.Validator, []uint64) {
|
|
balances := []uint64{0, 2, 4, 6, 8, 10, 12, 14, 16, 18}
|
|
return generateTestValidators(10,
|
|
allValidatorsValid(types.Slot(99)),
|
|
balanceIsKeyTimes2), balances
|
|
}
|
|
|
|
func TestStateBalanceCache(t *testing.T) {
|
|
type sbcTestCase struct {
|
|
err error
|
|
root [32]byte
|
|
sbc *stateBalanceCache
|
|
balances []uint64
|
|
name string
|
|
}
|
|
sentinelCacheMiss := errors.New("cache missed, as expected")
|
|
sentinelBalances := []uint64{1, 2, 3, 4, 5}
|
|
halfExpiredValidators, halfExpiredBalances := testHalfExpiredValidators()
|
|
halfQueuedValidators, halfQueuedBalances := testHalfQueuedValidators()
|
|
allValidValidators, allValidBalances := testAllValidValidators()
|
|
cases := []sbcTestCase{
|
|
{
|
|
root: bytesutil.ToBytes32([]byte{'A'}),
|
|
balances: sentinelBalances,
|
|
sbc: &stateBalanceCache{
|
|
stateGen: &mockStateByRooter{
|
|
err: sentinelCacheMiss,
|
|
},
|
|
root: bytesutil.ToBytes32([]byte{'A'}),
|
|
balances: sentinelBalances,
|
|
},
|
|
name: "cache hit",
|
|
},
|
|
// this works by using a staterooter that returns a known error
|
|
// so really we're testing the miss by making sure stategen got called
|
|
// this also tells us stategen errors are propagated
|
|
{
|
|
sbc: &stateBalanceCache{
|
|
stateGen: &mockStateByRooter{
|
|
err: sentinelCacheMiss,
|
|
},
|
|
root: bytesutil.ToBytes32([]byte{'B'}),
|
|
},
|
|
err: sentinelCacheMiss,
|
|
root: bytesutil.ToBytes32([]byte{'A'}),
|
|
name: "cache miss",
|
|
},
|
|
{
|
|
sbc: &stateBalanceCache{
|
|
stateGen: &mockStateByRooter{},
|
|
root: bytesutil.ToBytes32([]byte{'B'}),
|
|
},
|
|
err: errNilStateFromStategen,
|
|
root: bytesutil.ToBytes32([]byte{'A'}),
|
|
name: "error for nil state upon cache miss",
|
|
},
|
|
{
|
|
sbc: &stateBalanceCache{
|
|
stateGen: &mockStateByRooter{
|
|
state: testStateFixture(
|
|
testStateWithSlot(99),
|
|
testStateWithValidators(halfExpiredValidators)),
|
|
},
|
|
},
|
|
balances: halfExpiredBalances,
|
|
root: bytesutil.ToBytes32([]byte{'A'}),
|
|
name: "test filtering by exit epoch",
|
|
},
|
|
{
|
|
sbc: &stateBalanceCache{
|
|
stateGen: &mockStateByRooter{
|
|
state: testStateFixture(
|
|
testStateWithSlot(99),
|
|
testStateWithValidators(halfQueuedValidators)),
|
|
},
|
|
},
|
|
balances: halfQueuedBalances,
|
|
root: bytesutil.ToBytes32([]byte{'A'}),
|
|
name: "test filtering by activation epoch",
|
|
},
|
|
{
|
|
sbc: &stateBalanceCache{
|
|
stateGen: &mockStateByRooter{
|
|
state: testStateFixture(
|
|
testStateWithSlot(99),
|
|
testStateWithValidators(allValidValidators)),
|
|
},
|
|
},
|
|
balances: allValidBalances,
|
|
root: bytesutil.ToBytes32([]byte{'A'}),
|
|
name: "happy path",
|
|
},
|
|
{
|
|
sbc: &stateBalanceCache{
|
|
stateGen: &mockStateByRooter{
|
|
state: testStateFixture(
|
|
testStateWithSlot(99),
|
|
testStateWithValidators(allValidValidators)),
|
|
},
|
|
},
|
|
balances: allValidBalances,
|
|
root: [32]byte{},
|
|
name: "zero root",
|
|
},
|
|
}
|
|
ctx := context.Background()
|
|
for _, c := range cases {
|
|
t.Run(c.name, func(t *testing.T) {
|
|
cache := c.sbc
|
|
cacheRootStart := cache.root
|
|
b, err := cache.get(ctx, c.root)
|
|
require.ErrorIs(t, err, c.err)
|
|
require.DeepEqual(t, c.balances, b)
|
|
if c.err != nil {
|
|
// if there was an error somewhere, the root should not have changed (unless it already matched)
|
|
require.Equal(t, cacheRootStart, cache.root)
|
|
} else {
|
|
// when successful, the cache should always end with a root matching the request
|
|
require.Equal(t, c.root, cache.root)
|
|
}
|
|
})
|
|
}
|
|
}
|