package state_native import ( "context" "testing" "github.com/prysmaticlabs/prysm/v5/beacon-chain/state/state-native/types" "github.com/prysmaticlabs/prysm/v5/config/params" "github.com/prysmaticlabs/prysm/v5/consensus-types/primitives" ethpb "github.com/prysmaticlabs/prysm/v5/proto/prysm/v1alpha1" "github.com/prysmaticlabs/prysm/v5/testing/assert" "github.com/prysmaticlabs/prysm/v5/testing/require" ) func TestBeaconState_RotateAttestations(t *testing.T) { st, err := InitializeFromProtoPhase0(ðpb.BeaconState{ Slot: 1, CurrentEpochAttestations: []*ethpb.PendingAttestation{{Data: ðpb.AttestationData{Slot: 456}}}, PreviousEpochAttestations: []*ethpb.PendingAttestation{{Data: ðpb.AttestationData{Slot: 123}}}, }) require.NoError(t, err) require.NoError(t, st.RotateAttestations()) s, ok := st.(*BeaconState) require.Equal(t, true, ok) require.Equal(t, 0, len(s.currentEpochAttestationsVal())) require.Equal(t, primitives.Slot(456), s.previousEpochAttestationsVal()[0].Data.Slot) } func TestAppendBeyondIndicesLimit(t *testing.T) { zeroHash := params.BeaconConfig().ZeroHash mockblockRoots := make([][]byte, params.BeaconConfig().SlotsPerHistoricalRoot) for i := 0; i < len(mockblockRoots); i++ { mockblockRoots[i] = zeroHash[:] } mockstateRoots := make([][]byte, params.BeaconConfig().SlotsPerHistoricalRoot) for i := 0; i < len(mockstateRoots); i++ { mockstateRoots[i] = zeroHash[:] } mockrandaoMixes := make([][]byte, params.BeaconConfig().EpochsPerHistoricalVector) for i := 0; i < len(mockrandaoMixes); i++ { mockrandaoMixes[i] = zeroHash[:] } st, err := InitializeFromProtoPhase0(ðpb.BeaconState{ Slot: 1, CurrentEpochAttestations: []*ethpb.PendingAttestation{{Data: ðpb.AttestationData{Slot: 456}}}, PreviousEpochAttestations: []*ethpb.PendingAttestation{{Data: ðpb.AttestationData{Slot: 123}}}, Validators: []*ethpb.Validator{}, Eth1Data: ðpb.Eth1Data{}, BlockRoots: mockblockRoots, StateRoots: mockstateRoots, RandaoMixes: mockrandaoMixes, }) require.NoError(t, err) _, err = st.HashTreeRoot(context.Background()) require.NoError(t, err) s, ok := st.(*BeaconState) require.Equal(t, true, ok) for i := types.FieldIndex(0); i < types.FieldIndex(params.BeaconConfig().BeaconStateFieldCount); i++ { s.dirtyFields[i] = true } _, err = st.HashTreeRoot(context.Background()) require.NoError(t, err) for i := 0; i < 10; i++ { assert.NoError(t, st.AppendValidator(ðpb.Validator{})) } assert.Equal(t, false, s.rebuildTrie[types.Validators]) assert.NotEqual(t, len(s.dirtyIndices[types.Validators]), 0) for i := 0; i < indicesLimit; i++ { assert.NoError(t, st.AppendValidator(ðpb.Validator{})) } assert.Equal(t, true, s.rebuildTrie[types.Validators]) assert.Equal(t, len(s.dirtyIndices[types.Validators]), 0) } func BenchmarkAppendPreviousEpochAttestations(b *testing.B) { st, err := InitializeFromProtoPhase0(ðpb.BeaconState{}) require.NoError(b, err) max := params.BeaconConfig().PreviousEpochAttestationsLength() if max < 2 { b.Fatalf("previous epoch attestations length is less than 2: %d", max) } for i := uint64(0); i < max-2; i++ { err := st.AppendPreviousEpochAttestations(ðpb.PendingAttestation{Data: ðpb.AttestationData{Slot: primitives.Slot(i)}}) require.NoError(b, err) } b.ResetTimer() ref := st.Copy() for i := 0; i < b.N; i++ { err := ref.AppendPreviousEpochAttestations(ðpb.PendingAttestation{Data: ðpb.AttestationData{Slot: primitives.Slot(i)}}) require.NoError(b, err) ref = st.Copy() } }