mirror of
https://github.com/OffchainLabs/prysm.git
synced 2026-01-09 23:48:06 -05:00
* Electra: Beacon State * Electra: Beacon state fixes from PR 13919 * Add missing tests - part 1 * Split eip_7251_root.go into different files and reuse/share code with historical state summaries root. It's identical! * Add missing tests - part 2 * deposit receipts start index getters and setters (#13947) * adding in getters and setters for deposit receipts start index * adding tests * gaz * Add missing tests - part 3 of 3 Update the electra withdrawal example with a ssz state containing pending partial withdrawals * add tests for beacon-chain/state/state-native/getters_balance_deposits.go * Add electra field to testing/util/block.go execution payload * godoc commentary on public methods * Fix failing test * Add balances index out of bounds check and relevant tests. * Revert switch case electra * Instead of copying spectest data into testdata, use the spectest dependency * Deepsource fixes * Address @rkapka PR feedback * s/MaxPendingPartialsPerWithdrawalSweep/MaxPendingPartialsPerWithdrawalsSweep/ * Use multivalue slice compatible accessors for validator and balance in ActiveBalanceAtIndex * More @rkapka feedback. What a great reviewer! * More tests for branching logic in ExitEpochAndUpdateChurn * fix build --------- Co-authored-by: james-prysm <90280386+james-prysm@users.noreply.github.com>
201 lines
6.6 KiB
Go
201 lines
6.6 KiB
Go
package state_native_test
|
|
|
|
import (
|
|
"testing"
|
|
|
|
"github.com/golang/snappy"
|
|
"github.com/prysmaticlabs/prysm/v5/beacon-chain/core/helpers"
|
|
state_native "github.com/prysmaticlabs/prysm/v5/beacon-chain/state/state-native"
|
|
"github.com/prysmaticlabs/prysm/v5/config/params"
|
|
"github.com/prysmaticlabs/prysm/v5/consensus-types/primitives"
|
|
"github.com/prysmaticlabs/prysm/v5/math"
|
|
eth "github.com/prysmaticlabs/prysm/v5/proto/prysm/v1alpha1"
|
|
"github.com/prysmaticlabs/prysm/v5/testing/require"
|
|
"github.com/prysmaticlabs/prysm/v5/testing/util"
|
|
"github.com/prysmaticlabs/prysm/v5/time/slots"
|
|
)
|
|
|
|
func TestExitEpochAndUpdateChurn_SpectestCase(t *testing.T) {
|
|
// Load a serialized Electra state from disk.
|
|
// The spec tests shows that the exit epoch is 262 for validator 0 performing a voluntary exit.
|
|
serializedBytes, err := util.BazelFileBytes("tests/mainnet/electra/operations/voluntary_exit/pyspec_tests/exit_existing_churn_and_churn_limit_balance/pre.ssz_snappy")
|
|
require.NoError(t, err)
|
|
serializedSSZ, err := snappy.Decode(nil /* dst */, serializedBytes)
|
|
require.NoError(t, err)
|
|
pb := ð.BeaconStateElectra{}
|
|
require.NoError(t, pb.UnmarshalSSZ(serializedSSZ))
|
|
s, err := state_native.InitializeFromProtoElectra(pb)
|
|
require.NoError(t, err)
|
|
|
|
val, err := s.ValidatorAtIndex(0)
|
|
require.NoError(t, err)
|
|
|
|
ee, err := s.ExitEpochAndUpdateChurn(math.Gwei(val.EffectiveBalance))
|
|
require.NoError(t, err)
|
|
require.Equal(t, primitives.Epoch(262), ee)
|
|
|
|
p := s.ToProto()
|
|
pb, ok := p.(*eth.BeaconStateElectra)
|
|
if !ok {
|
|
t.Fatal("wrong proto")
|
|
}
|
|
require.Equal(t, math.Gwei(127000000000), pb.ExitBalanceToConsume)
|
|
require.Equal(t, primitives.Epoch(262), pb.EarliestExitEpoch)
|
|
|
|
// Fails for versions older than electra
|
|
s, err = state_native.InitializeFromProtoDeneb(ð.BeaconStateDeneb{})
|
|
require.NoError(t, err)
|
|
_, err = s.ExitEpochAndUpdateChurn(10)
|
|
require.ErrorContains(t, "not supported", err)
|
|
}
|
|
|
|
func TestExitEpochAndUpdateChurn(t *testing.T) {
|
|
slot := primitives.Slot(10_000_000)
|
|
epoch := slots.ToEpoch(slot)
|
|
t.Run("state earliest exit epoch is old", func(t *testing.T) {
|
|
st, err := state_native.InitializeFromProtoElectra(ð.BeaconStateElectra{
|
|
Slot: slot,
|
|
Validators: []*eth.Validator{
|
|
{
|
|
EffectiveBalance: params.BeaconConfig().MaxEffectiveBalanceElectra,
|
|
},
|
|
},
|
|
Balances: []uint64{params.BeaconConfig().MaxEffectiveBalanceElectra},
|
|
EarliestExitEpoch: epoch - params.BeaconConfig().MaxSeedLookahead*2, // Old, relative to slot.
|
|
ExitBalanceToConsume: math.Gwei(20_000_000),
|
|
})
|
|
require.NoError(t, err)
|
|
activeBal, err := helpers.TotalActiveBalance(st)
|
|
require.NoError(t, err)
|
|
|
|
exitBal := math.Gwei(10_000_000)
|
|
|
|
wantExitBalToConsume := helpers.ActivationExitChurnLimit(math.Gwei(activeBal)) - exitBal
|
|
|
|
ee, err := st.ExitEpochAndUpdateChurn(exitBal)
|
|
require.NoError(t, err)
|
|
|
|
wantExitEpoch := helpers.ActivationExitEpoch(epoch)
|
|
require.Equal(t, wantExitEpoch, ee)
|
|
|
|
p := st.ToProto()
|
|
pb, ok := p.(*eth.BeaconStateElectra)
|
|
if !ok {
|
|
t.Fatal("wrong proto")
|
|
}
|
|
require.Equal(t, wantExitBalToConsume, pb.ExitBalanceToConsume)
|
|
require.Equal(t, wantExitEpoch, pb.EarliestExitEpoch)
|
|
})
|
|
|
|
t.Run("state exit bal to consume is less than activation exit churn limit", func(t *testing.T) {
|
|
st, err := state_native.InitializeFromProtoElectra(ð.BeaconStateElectra{
|
|
Slot: slot,
|
|
Validators: []*eth.Validator{
|
|
{
|
|
EffectiveBalance: params.BeaconConfig().MaxEffectiveBalanceElectra,
|
|
},
|
|
},
|
|
Balances: []uint64{params.BeaconConfig().MaxEffectiveBalanceElectra},
|
|
EarliestExitEpoch: epoch,
|
|
ExitBalanceToConsume: math.Gwei(20_000_000),
|
|
})
|
|
require.NoError(t, err)
|
|
activeBal, err := helpers.TotalActiveBalance(st)
|
|
require.NoError(t, err)
|
|
|
|
activationExitChurnLimit := helpers.ActivationExitChurnLimit(math.Gwei(activeBal))
|
|
exitBal := activationExitChurnLimit * 2
|
|
|
|
wantExitBalToConsume := math.Gwei(0)
|
|
|
|
ee, err := st.ExitEpochAndUpdateChurn(exitBal)
|
|
require.NoError(t, err)
|
|
|
|
wantExitEpoch := helpers.ActivationExitEpoch(epoch) + 1
|
|
require.Equal(t, wantExitEpoch, ee)
|
|
|
|
p := st.ToProto()
|
|
pb, ok := p.(*eth.BeaconStateElectra)
|
|
if !ok {
|
|
t.Fatal("wrong proto")
|
|
}
|
|
require.Equal(t, wantExitBalToConsume, pb.ExitBalanceToConsume)
|
|
require.Equal(t, wantExitEpoch, pb.EarliestExitEpoch)
|
|
})
|
|
|
|
t.Run("state earliest exit epoch is in the future and exit balance is less than state", func(t *testing.T) {
|
|
st, err := state_native.InitializeFromProtoElectra(ð.BeaconStateElectra{
|
|
Slot: slot,
|
|
Validators: []*eth.Validator{
|
|
{
|
|
EffectiveBalance: params.BeaconConfig().MaxEffectiveBalanceElectra,
|
|
},
|
|
},
|
|
Balances: []uint64{params.BeaconConfig().MaxEffectiveBalanceElectra},
|
|
EarliestExitEpoch: epoch + 10_000,
|
|
ExitBalanceToConsume: math.Gwei(20_000_000),
|
|
})
|
|
require.NoError(t, err)
|
|
|
|
exitBal := math.Gwei(10_000_000)
|
|
|
|
wantExitBalToConsume := math.Gwei(20_000_000) - exitBal
|
|
|
|
ee, err := st.ExitEpochAndUpdateChurn(exitBal)
|
|
require.NoError(t, err)
|
|
|
|
wantExitEpoch := epoch + 10_000
|
|
require.Equal(t, wantExitEpoch, ee)
|
|
|
|
p := st.ToProto()
|
|
pb, ok := p.(*eth.BeaconStateElectra)
|
|
if !ok {
|
|
t.Fatal("wrong proto")
|
|
}
|
|
require.Equal(t, wantExitBalToConsume, pb.ExitBalanceToConsume)
|
|
require.Equal(t, wantExitEpoch, pb.EarliestExitEpoch)
|
|
})
|
|
|
|
t.Run("state earliest exit epoch is in the future and exit balance exceeds state", func(t *testing.T) {
|
|
st, err := state_native.InitializeFromProtoElectra(ð.BeaconStateElectra{
|
|
Slot: slot,
|
|
Validators: []*eth.Validator{
|
|
{
|
|
EffectiveBalance: params.BeaconConfig().MaxEffectiveBalanceElectra,
|
|
},
|
|
},
|
|
Balances: []uint64{params.BeaconConfig().MaxEffectiveBalanceElectra},
|
|
EarliestExitEpoch: epoch + 10_000,
|
|
ExitBalanceToConsume: math.Gwei(20_000_000),
|
|
})
|
|
require.NoError(t, err)
|
|
|
|
exitBal := math.Gwei(40_000_000)
|
|
activeBal, err := helpers.TotalActiveBalance(st)
|
|
require.NoError(t, err)
|
|
activationExitChurnLimit := helpers.ActivationExitChurnLimit(math.Gwei(activeBal))
|
|
wantExitBalToConsume := activationExitChurnLimit - 20_000_000
|
|
|
|
ee, err := st.ExitEpochAndUpdateChurn(exitBal)
|
|
require.NoError(t, err)
|
|
|
|
wantExitEpoch := epoch + 10_000 + 1
|
|
require.Equal(t, wantExitEpoch, ee)
|
|
|
|
p := st.ToProto()
|
|
pb, ok := p.(*eth.BeaconStateElectra)
|
|
if !ok {
|
|
t.Fatal("wrong proto")
|
|
}
|
|
require.Equal(t, wantExitBalToConsume, pb.ExitBalanceToConsume)
|
|
require.Equal(t, wantExitEpoch, pb.EarliestExitEpoch)
|
|
})
|
|
|
|
t.Run("earlier than electra returns error", func(t *testing.T) {
|
|
st, err := state_native.InitializeFromProtoDeneb(ð.BeaconStateDeneb{})
|
|
require.NoError(t, err)
|
|
_, err = st.ExitEpochAndUpdateChurn(0)
|
|
require.ErrorContains(t, "is not supported", err)
|
|
})
|
|
}
|