Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(blockchain): introducing validator size cap size #2119

Merged
merged 102 commits into from
Dec 2, 2024
Merged
Show file tree
Hide file tree
Changes from 79 commits
Commits
Show all changes
102 commits
Select commit Hold shift + click to select a range
b4a1077
skip zero amount withdrawals to reduce number of withdrawals processed
abi87 Oct 28, 2024
f4a7d79
wip: adding UTs to state transition package
abi87 Oct 29, 2024
0f46172
wip: completed simple UT for state transition package
abi87 Oct 29, 2024
22c7717
wip: minimal execution engine stub
abi87 Oct 29, 2024
05cba80
extended asserts
abi87 Oct 29, 2024
443ac1b
added test case
abi87 Oct 29, 2024
10efd5e
nits
abi87 Oct 29, 2024
099716d
tests for helpers in state transition using mock
nidhi-singh02 Oct 30, 2024
151a533
Revert "tests for helpers in state transition using mock"
nidhi-singh02 Oct 30, 2024
9818c7e
tests with only mock for execution engine
nidhi-singh02 Oct 30, 2024
160cc88
removed test for VerifyAndNotifyNewPayload
nidhi-singh02 Oct 30, 2024
4a9fe1c
nit
abi87 Oct 31, 2024
e048be4
improved unit tests asserts
abi87 Oct 31, 2024
8bf34db
appease linter
abi87 Oct 31, 2024
d90a95a
fix(state-transition): fix deposit index upon genesis processing (#2116)
abi87 Oct 31, 2024
e17d29c
fixed bad merge
abi87 Oct 31, 2024
8dfe59d
Merge branch 'main' into fix-maximum-number-withdrawals
abi87 Nov 1, 2024
daa9262
Merge branch 'state-transition-add-UTs' into fix-maximum-number-withd…
abi87 Nov 1, 2024
92d494f
added UTs to show no withdrawals are made when unnecessary
abi87 Nov 1, 2024
6286b20
Merge branch 'main' into state-transition-add-UTs
abi87 Nov 1, 2024
a1a3def
Merge branch 'state-transition-add-UTs' into fix-maximum-number-withd…
abi87 Nov 1, 2024
b395ac2
added validator set cap size to chain specs
abi87 Nov 1, 2024
63e4db0
Merge branch 'fix-maximum-number-withdrawals' into validator-size-cap
abi87 Nov 1, 2024
36fe774
enforce validator set size cap on genesis
abi87 Nov 1, 2024
e64bc74
appease gosec
abi87 Nov 1, 2024
c6c68c3
mark extra validators as withdrawable
abi87 Nov 1, 2024
37542a0
nit
abi87 Nov 1, 2024
ea69a35
avoid double validators set cap checks on genesis
abi87 Nov 1, 2024
af43b93
add UT showing extra validator is added ready for withdrawal
abi87 Nov 1, 2024
a728bcb
extend UT to show extra validator deposit is withdrawn
abi87 Nov 1, 2024
97fba5c
fix withdrawal index update
abi87 Nov 1, 2024
826351d
Merge branch 'fix-maximum-number-withdrawals' into validator-size-cap
abi87 Nov 1, 2024
af8c5e0
fix(build): erigon repo
gummybera Nov 1, 2024
023ebfd
fix(build): bump erigon to recent version
gummybera Nov 1, 2024
d66b298
nits from code review
abi87 Nov 1, 2024
43b1eb4
Merge branch 'fix-erigon' into validator-size-cap
abi87 Nov 1, 2024
420f621
Merge branch 'state-transition-add-UTs' into fix-maximum-number-withd…
abi87 Nov 1, 2024
561f95f
Merge branch 'fix-maximum-number-withdrawals' into validator-size-cap
abi87 Nov 2, 2024
09eee8e
Merge branch 'fix-erigon' into fix-maximum-number-withdrawals
abi87 Nov 2, 2024
72539e0
Merge branch 'fix-maximum-number-withdrawals' into validator-size-cap
abi87 Nov 4, 2024
d8a267e
hacked fix for withdrawals SSZ serialization
abi87 Nov 4, 2024
3603f6b
Merge branch 'fix-maximum-number-withdrawals' into validator-size-cap
abi87 Nov 4, 2024
32fbbbc
nits
abi87 Nov 4, 2024
ebff958
nit
abi87 Nov 6, 2024
8c9cf2d
Merge branch 'main' into validator-size-cap
abi87 Nov 6, 2024
f51b157
renamed GetValidatorSetCapSize
abi87 Nov 6, 2024
5e38855
nits
abi87 Nov 6, 2024
bc26a85
wip: evicting smallest validator rathen than latest
abi87 Nov 6, 2024
401ab1a
improved UTs
abi87 Nov 6, 2024
9100d91
leftover
abi87 Nov 6, 2024
3699c20
Merge branch 'main' into validator-size-cap
abi87 Nov 6, 2024
9a7c057
consolidated UTs
abi87 Nov 7, 2024
0892421
minor test helpers renaming
abi87 Nov 7, 2024
1bd5f95
Merge branch 'main' into validator-size-cap
abi87 Nov 7, 2024
174229a
improving checks on validator set cap
abi87 Nov 7, 2024
3cca1f3
added processEffectiveBalanceUpdates
abi87 Nov 7, 2024
0ae0c7d
updated chain specs
abi87 Nov 7, 2024
3df7fc7
nits
abi87 Nov 7, 2024
4d3de74
nit
abi87 Nov 7, 2024
77f49c6
further UT extension
abi87 Nov 7, 2024
b96f776
fixed typo
abi87 Nov 7, 2024
882b91e
added processEffectiveBalanceUpdates
abi87 Nov 8, 2024
bca903b
Merge branch 'complete-validator-set-updates' into validator-size-cap
abi87 Nov 8, 2024
ebcf8b4
wip: update validator set returned to consensus
abi87 Nov 8, 2024
d4ec3e7
nit
abi87 Nov 8, 2024
3245f96
nit
abi87 Nov 8, 2024
bfca971
added cases for genesis initialization UTs
abi87 Nov 8, 2024
5d5a57d
Merge branch 'main' into complete-validator-set-updates
abi87 Nov 9, 2024
b898788
made process deposits closer to eth 2.0 specs
abi87 Nov 9, 2024
34ea814
nit
abi87 Nov 9, 2024
e2962ac
fixed expectations in UTs
abi87 Nov 9, 2024
0cdb569
fixed validators balance update
abi87 Nov 9, 2024
3f3d4eb
nits
abi87 Nov 9, 2024
3d9e3ab
fixed faulty assertion in UTs
abi87 Nov 9, 2024
0183914
added UT for validator creation
abi87 Nov 9, 2024
eda7909
minor processSyncCommitteeUpdates renaming
abi87 Nov 9, 2024
3e734e0
Merge branch 'complete-validator-set-updates' into validator-size-cap
abi87 Nov 9, 2024
c6b5586
send only validators set diffs
abi87 Nov 9, 2024
e9f01e2
minor UT code consolidation
abi87 Nov 10, 2024
3bc7678
appease linter
abi87 Nov 10, 2024
679da51
nits
abi87 Nov 10, 2024
b6d62c5
minor readme
abi87 Nov 10, 2024
f361fe4
Merge branch 'complete-validator-set-updates' into validator-set-upda…
abi87 Nov 10, 2024
260bddb
nit
abi87 Nov 10, 2024
f3c2b50
README nit
abi87 Nov 10, 2024
013aa95
Merge branch 'complete-validator-set-updates' into validator-size-cap
abi87 Nov 10, 2024
9b139f8
added UT for double eviction
abi87 Nov 10, 2024
0cbd981
fixed double evictions
abi87 Nov 10, 2024
52c4689
fixed eviction epoch
abi87 Nov 11, 2024
80af103
appease rabbit
abi87 Nov 11, 2024
976c27b
Merge branch 'validator-set-updates-optimizations' into validator-siz…
abi87 Nov 11, 2024
1cd96e8
cleanup UTs and improved asserts
abi87 Nov 11, 2024
e8db0d9
Merge branch 'main' into validator-size-cap
abi87 Nov 28, 2024
5ff81ac
Merge branch 'main' into validator-size-cap
abi87 Nov 28, 2024
4774160
Merge branch 'validator-size-cap' of https://github.com/berachain/bea…
abi87 Nov 29, 2024
78d6429
nit
abi87 Nov 29, 2024
e8b4dfd
Merge branch 'main' into validator-size-cap
abi87 Dec 2, 2024
4f90bfe
nit
abi87 Dec 2, 2024
f36db52
nit renaming + validation of valSetCap chain spec
calbera Dec 2, 2024
39b83ca
boonet TODOs for values
calbera Dec 2, 2024
355d99b
appease linter
abi87 Dec 2, 2024
0307a03
nit renaming error val set cap exceeded
calbera Dec 2, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
43 changes: 43 additions & 0 deletions mod/chain-spec/pkg/chain/chain_spec.go
Original file line number Diff line number Diff line change
Expand Up @@ -45,6 +45,19 @@ type Spec[
// calculations.
EffectiveBalanceIncrement() uint64

// HysteresisQuotient returns the quotient used in effective balance
// calculations to create hysteresis. This provides resistance to small
// balance changes triggering effective balance updates.
HysteresisQuotient() uint64

// HysteresisDownwardMultiplier returns the multiplier used when checking
// if the effective balance should be decreased.
HysteresisDownwardMultiplier() uint64

// HysteresisUpwardMultiplier returns the multiplier used when checking
// if the effective balance should be increased.
HysteresisUpwardMultiplier() uint64

// Time parameters constants.

// SlotsPerEpoch returns the number of slots in an epoch.
Expand Down Expand Up @@ -183,6 +196,10 @@ type Spec[
// GetCometBFTConfigForSlot retrieves the CometBFT config for a specific
// slot.
GetCometBFTConfigForSlot(slot SlotT) CometBFTConfigT

// GetValidatorSetCapSize retrieves the maximum number of validators
// allowed in the active set.
GetValidatorSetCapSize() uint32
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🛠️ Refactor suggestion

Rename method to GetValidatorSetCap for consistency.

Based on previous review discussions, the method should be renamed while keeping the uint32 return type.

-       // GetValidatorSetCapSize retrieves the maximum number of validators
+       // GetValidatorSetCap retrieves the maximum number of validators
        // allowed in the active set.
-       GetValidatorSetCapSize() uint32
+       GetValidatorSetCap() uint32
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
// GetValidatorSetCapSize retrieves the maximum number of validators
// allowed in the active set.
GetValidatorSetCapSize() uint32
// GetValidatorSetCap retrieves the maximum number of validators
// allowed in the active set.
GetValidatorSetCap() uint32

}

// chainSpec is a concrete implementation of the ChainSpec interface, holding
Expand Down Expand Up @@ -245,6 +262,24 @@ func (c chainSpec[
return c.Data.EffectiveBalanceIncrement
}

func (c chainSpec[
DomainTypeT, EpochT, ExecutionAddressT, SlotT, CometBFTConfigT,
]) HysteresisQuotient() uint64 {
return c.Data.HysteresisQuotient
}

func (c chainSpec[
DomainTypeT, EpochT, ExecutionAddressT, SlotT, CometBFTConfigT,
]) HysteresisDownwardMultiplier() uint64 {
return c.Data.HysteresisDownwardMultiplier
}

func (c chainSpec[
DomainTypeT, EpochT, ExecutionAddressT, SlotT, CometBFTConfigT,
]) HysteresisUpwardMultiplier() uint64 {
return c.Data.HysteresisUpwardMultiplier
}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🧹 Nitpick (assertive)

Add documentation to implementation methods.

The implementation methods should include documentation comments for consistency with other methods in the file.

+       // HysteresisQuotient returns the quotient used in hysteresis calculations.
        func (c chainSpec[
                DomainTypeT, EpochT, ExecutionAddressT, SlotT, CometBFTConfigT,
        ]) HysteresisQuotient() uint64 {
                return c.Data.HysteresisQuotient
        }

+       // HysteresisDownwardMultiplier returns the multiplier used when adjusting values downward.
        func (c chainSpec[
                DomainTypeT, EpochT, ExecutionAddressT, SlotT, CometBFTConfigT,
        ]) HysteresisDownwardMultiplier() uint64 {
                return c.Data.HysteresisDownwardMultiplier
        }

+       // HysteresisUpwardMultiplier returns the multiplier used when adjusting values upward.
        func (c chainSpec[
                DomainTypeT, EpochT, ExecutionAddressT, SlotT, CometBFTConfigT,
        ]) HysteresisUpwardMultiplier() uint64 {
                return c.Data.HysteresisUpwardMultiplier
        }
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
func (c chainSpec[
DomainTypeT, EpochT, ExecutionAddressT, SlotT, CometBFTConfigT,
]) HysteresisQuotient() uint64 {
return c.Data.HysteresisQuotient
}
func (c chainSpec[
DomainTypeT, EpochT, ExecutionAddressT, SlotT, CometBFTConfigT,
]) HysteresisDownwardMultiplier() uint64 {
return c.Data.HysteresisDownwardMultiplier
}
func (c chainSpec[
DomainTypeT, EpochT, ExecutionAddressT, SlotT, CometBFTConfigT,
]) HysteresisUpwardMultiplier() uint64 {
return c.Data.HysteresisUpwardMultiplier
}
// HysteresisQuotient returns the quotient used in hysteresis calculations.
func (c chainSpec[
DomainTypeT, EpochT, ExecutionAddressT, SlotT, CometBFTConfigT,
]) HysteresisQuotient() uint64 {
return c.Data.HysteresisQuotient
}
// HysteresisDownwardMultiplier returns the multiplier used when adjusting values downward.
func (c chainSpec[
DomainTypeT, EpochT, ExecutionAddressT, SlotT, CometBFTConfigT,
]) HysteresisDownwardMultiplier() uint64 {
return c.Data.HysteresisDownwardMultiplier
}
// HysteresisUpwardMultiplier returns the multiplier used when adjusting values upward.
func (c chainSpec[
DomainTypeT, EpochT, ExecutionAddressT, SlotT, CometBFTConfigT,
]) HysteresisUpwardMultiplier() uint64 {
return c.Data.HysteresisUpwardMultiplier
}


// SlotsPerEpoch returns the number of slots per epoch.
func (c chainSpec[
DomainTypeT, EpochT, ExecutionAddressT, SlotT, CometBFTConfigT,
Expand Down Expand Up @@ -476,3 +511,11 @@ func (c chainSpec[
]) GetCometBFTConfigForSlot(_ SlotT) CometBFTConfigT {
return c.Data.CometValues
}

// GetValidatorSetCapSize retrieves the maximum number of validators
// allowed in the active set.
func (c chainSpec[
abi87 marked this conversation as resolved.
Show resolved Hide resolved
DomainTypeT, EpochT, ExecutionAddressT, SlotT, CometBFTConfigT,
]) GetValidatorSetCapSize() uint32 {
return c.Data.ValidatorSetCapSize
}
abi87 marked this conversation as resolved.
Show resolved Hide resolved
10 changes: 10 additions & 0 deletions mod/chain-spec/pkg/chain/data.go
Original file line number Diff line number Diff line change
Expand Up @@ -43,6 +43,13 @@ type SpecData[
// EffectiveBalanceIncrement is the effective balance increment.
EffectiveBalanceIncrement uint64 `mapstructure:"effective-balance-increment"`

// HysteresisQuotient is the quotient used in effective balance calculations
HysteresisQuotient uint64 `mapstructure:"hysteresis-quotient"`
// HysteresisDownwardMultiplier is the multiplier for downward balance adjustments.
HysteresisDownwardMultiplier uint64 `mapstructure:"hysteresis-downward-multiplier"`
// HysteresisUpwardMultiplier is the multiplier for upward balance adjustments.
HysteresisUpwardMultiplier uint64 `mapstructure:"hysteresis-upward-multiplier"`

// Time parameters constants.
//
// SlotsPerEpoch is the number of slots per epoch.
Expand Down Expand Up @@ -146,4 +153,7 @@ type SpecData[

// CometValues
CometValues CometBFTConfigT `mapstructure:"comet-bft-config"`

// Validators Set config
ValidatorSetCapSize uint32 `mapstructure:"validator-set-cap-size"`
abi87 marked this conversation as resolved.
Show resolved Hide resolved
abi87 marked this conversation as resolved.
Show resolved Hide resolved
abi87 marked this conversation as resolved.
Show resolved Hide resolved
}
12 changes: 8 additions & 4 deletions mod/config/pkg/spec/testnet.go
Original file line number Diff line number Diff line change
Expand Up @@ -60,10 +60,13 @@ func BaseSpec() chain.SpecData[
any,
]{
// // Gwei value constants.
MinDepositAmount: uint64(1e9),
MaxEffectiveBalance: uint64(32e9),
EjectionBalance: uint64(16e9),
EffectiveBalanceIncrement: uint64(1e9),
MinDepositAmount: uint64(1e9),
MaxEffectiveBalance: uint64(32e9),
EjectionBalance: uint64(16e9),
EffectiveBalanceIncrement: uint64(1e9),
HysteresisQuotient: uint64(4),
HysteresisDownwardMultiplier: uint64(1),
HysteresisUpwardMultiplier: uint64(5),
// Time parameters constants.
SlotsPerEpoch: 32,
MinEpochsToInactivityPenalty: 4,
Expand Down Expand Up @@ -123,5 +126,6 @@ func BaseSpec() chain.SpecData[
BytesPerBlob: 131072,
KZGCommitmentInclusionProofDepth: 17,
CometValues: cmtConsensusParams,
ValidatorSetCapSize: 256,
}
}
5 changes: 5 additions & 0 deletions mod/consensus-types/pkg/types/validator.go
Original file line number Diff line number Diff line change
Expand Up @@ -325,6 +325,11 @@ func (v *Validator) SetEffectiveBalance(balance math.Gwei) {
v.EffectiveBalance = balance
}

// SetWithdrawableEpoch sets the epoch when the validator can withdraw.
func (v *Validator) SetWithdrawableEpoch(e math.Epoch) {
abi87 marked this conversation as resolved.
Show resolved Hide resolved
v.WithdrawableEpoch = e
}
calbera marked this conversation as resolved.
Show resolved Hide resolved

// GetWithdrawableEpoch returns the epoch when the validator can withdraw.
func (v Validator) GetWithdrawableEpoch() math.Epoch {
return v.WithdrawableEpoch
Expand Down
11 changes: 11 additions & 0 deletions mod/state-transition/pkg/core/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
# State Processor

## Validators handling

Currently:

- Any validator whose effective balance is above `EjectionBalance` will stay a validator forever, as we have not (yet) implemented withdrawals facilities.
- Withdrawals are automatically generated only if a validator effective balance goes beyond `MaxEffectiveBalance`. In this case enough balance is scheduled for withdrawal, just enough to make validator's effective balance equal to `MaxEffectiveBalance`. Since `MaxEffectiveBalance` > `EjectionBalance`, the validator will keep being a validator.
- If a deposit is made for a validator with a balance smaller or equal to `EjectionBalance`, no validator will be created[^1] because of the insufficient balance. However currently the whole deposited balance is **not** scheduled for withdrawal at the next epoch.

[^1]: Technically a validator is made in the BeaconKit state to track the deposit, but such a validator is never returned to the consensus engine. Moreover the deposit should be evicted at the next epoch.
6 changes: 6 additions & 0 deletions mod/state-transition/pkg/core/errors.go
Original file line number Diff line number Diff line change
Expand Up @@ -23,13 +23,19 @@ package core
import "github.com/berachain/beacon-kit/mod/errors"

var (
// ErrValidatorSetCapHit is returned when we try and add a validator
// that would breach validator set size cap.
ErrHitValidatorsSetCap = errors.New("hit validator set size cap")
calbera marked this conversation as resolved.
Show resolved Hide resolved

// ErrBlockSlotTooLow is returned when the block slot is too low.
ErrBlockSlotTooLow = errors.New("block slot too low")

// ErrSlotMismatch is returned when the slot in a block header does not
// match the expected value.
ErrSlotMismatch = errors.New("slot mismatch")

// ErrProposerMismatch is returned when block builder does not match
// with the proposer reported by consensus.
ErrProposerMismatch = errors.New("proposer key mismatch")

// ErrParentRootMismatch is returned when the parent root in an execution
Expand Down
17 changes: 17 additions & 0 deletions mod/state-transition/pkg/core/helpers_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -219,3 +219,20 @@ func buildNextBlock(
Body: nextBlkBody,
}
}

// ValUpdatesDiff returns elements of slice a that are not in b.
func ValUpdatesDiff(
a, b []*transition.ValidatorUpdate,
) []*transition.ValidatorUpdate {
mb := make(map[string]struct{}, len(b))
for _, v := range b {
mb[v.Pubkey.String()] = struct{}{}
}
abi87 marked this conversation as resolved.
Show resolved Hide resolved
var diff []*transition.ValidatorUpdate
for _, x := range a {
if _, found := mb[x.Pubkey.String()]; !found {
diff = append(diff, x)
}
}
return diff
}
abi87 marked this conversation as resolved.
Show resolved Hide resolved
110 changes: 73 additions & 37 deletions mod/state-transition/pkg/core/state_processor.go
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,7 @@ package core
import (
"bytes"

"github.com/berachain/beacon-kit/mod/consensus-types/pkg/types"
"github.com/berachain/beacon-kit/mod/errors"
"github.com/berachain/beacon-kit/mod/primitives/pkg/common"
"github.com/berachain/beacon-kit/mod/primitives/pkg/constants"
Expand Down Expand Up @@ -197,34 +198,26 @@ func (sp *StateProcessor[
]) ProcessSlots(
st BeaconStateT, slot math.Slot,
) (transition.ValidatorUpdates, error) {
var (
validatorUpdates transition.ValidatorUpdates
epochValidatorUpdates transition.ValidatorUpdates
)

var res transition.ValidatorUpdates
stateSlot, err := st.GetSlot()
if err != nil {
return nil, err
}

// Iterate until we are "caught up".
for ; stateSlot < slot; stateSlot++ {
// Process the slot
if err = sp.processSlot(st); err != nil {
return nil, err
}

// Process the Epoch Boundary.
boundary := (stateSlot.Unwrap()+1)%sp.cs.SlotsPerEpoch() == 0
if boundary {
if epochValidatorUpdates, err =
sp.processEpoch(st); err != nil {
var epochUpdates transition.ValidatorUpdates
if epochUpdates, err = sp.processEpoch(st); err != nil {
return nil, err
}
validatorUpdates = append(
validatorUpdates,
epochValidatorUpdates...,
)
res = append(res, epochUpdates...)
}

// We update on the state because we need to
Expand All @@ -234,7 +227,7 @@ func (sp *StateProcessor[
}
}

return validatorUpdates, nil
return res, nil
}

// processSlot is run when a slot is missed.
Expand Down Expand Up @@ -336,6 +329,9 @@ func (sp *StateProcessor[
if err := sp.processRewardsAndPenalties(st); err != nil {
return nil, err
}
if err := sp.processEffectiveBalanceUpdates(st); err != nil {
return nil, err
}
if err := sp.processSlashingsReset(st); err != nil {
return nil, err
}
Expand All @@ -362,13 +358,12 @@ func (sp *StateProcessor[
}
if blk.GetSlot() != slot {
return errors.Wrapf(
ErrSlotMismatch,
"expected: %d, got: %d",
ErrSlotMismatch, "expected: %d, got: %d",
slot, blk.GetSlot(),
)
}

// Verify the parent block root is correct.
// Verify that the block is newer than latest block header
latestBlockHeader, err := st.GetLatestBlockHeader()
if err != nil {
return err
Expand All @@ -380,14 +375,6 @@ func (sp *StateProcessor[
)
}

parentBlockRoot := latestBlockHeader.HashTreeRoot()
if parentBlockRoot != blk.GetParentBlockRoot() {
return errors.Wrapf(ErrParentRootMismatch,
"expected: %s, got: %s",
parentBlockRoot.String(), blk.GetParentBlockRoot().String(),
)
}

// Verify that proposer matches with what consensus declares as proposer
proposer, err := st.ValidatorByIndex(blk.GetProposerIndex())
if err != nil {
Expand All @@ -404,26 +391,24 @@ func (sp *StateProcessor[
)
}

// Check to make sure the proposer isn't slashed.
if proposer.IsSlashed() {
// Verify that the parent matches
parentBlockRoot := latestBlockHeader.HashTreeRoot()
if parentBlockRoot != blk.GetParentBlockRoot() {
return errors.Wrapf(
ErrSlashedProposer,
"index: %d",
blk.GetProposerIndex(),
ErrParentRootMismatch, "expected: %s, got: %s",
parentBlockRoot.String(), blk.GetParentBlockRoot().String(),
)
}

// Ensure the block is within the acceptable range.
// TODO: move this is in the wrong spot.
deposits := blk.GetBody().GetDeposits()
if uint64(len(deposits)) > sp.cs.MaxDepositsPerBlock() {
return errors.Wrapf(ErrExceedsBlockDepositLimit,
"expected: %d, got: %d",
sp.cs.MaxDepositsPerBlock(), len(deposits),
// Verify proposer is not slashed
if proposer.IsSlashed() {
return errors.Wrapf(
ErrSlashedProposer, "index: %d",
blk.GetProposerIndex(),
)
}

// Calculate the body root to place on the header.
// Cache current block as the new latest block
bodyRoot := blk.GetBody().HashTreeRoot()
var lbh BeaconBlockHeaderT
lbh = lbh.New(
Expand Down Expand Up @@ -516,3 +501,54 @@ func (sp *StateProcessor[

return nil
}

// processEffectiveBalanceUpdates as defined in the Ethereum 2.0 specification.
// https://github.com/ethereum/consensus-specs/blob/dev/specs/phase0/beacon-chain.md#effective-balances-updates
//
//nolint:lll
func (sp *StateProcessor[
_, _, _, BeaconStateT, _, _, _, _, _, _, _, _, _, _, _, _, _,
]) processEffectiveBalanceUpdates(
st BeaconStateT,
) error {
// Update effective balances with hysteresis
validators, err := st.GetValidators()
if err != nil {
return err
}

var (
hysteresisIncrement = sp.cs.EffectiveBalanceIncrement() / sp.cs.HysteresisQuotient()
downwardThreshold = math.Gwei(hysteresisIncrement * sp.cs.HysteresisDownwardMultiplier())
upwardThreshold = math.Gwei(hysteresisIncrement * sp.cs.HysteresisUpwardMultiplier())

idx math.U64
balance math.Gwei
)

for _, val := range validators {
idx, err = st.ValidatorIndexByPubkey(val.GetPubkey())
if err != nil {
return err
}

balance, err = st.GetBalance(idx)
if err != nil {
return err
}

if balance+downwardThreshold < val.GetEffectiveBalance() ||
val.GetEffectiveBalance()+upwardThreshold < balance {
updatedBalance := types.ComputeEffectiveBalance(
balance,
math.U64(sp.cs.EffectiveBalanceIncrement()),
math.U64(sp.cs.MaxEffectiveBalance()),
)
val.SetEffectiveBalance(updatedBalance)
if err = st.UpdateValidatorAtIndex(idx, val); err != nil {
return err
}
}
}
return nil
}
13 changes: 12 additions & 1 deletion mod/state-transition/pkg/core/state_processor_committee.go
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@
package core

import (
"github.com/berachain/beacon-kit/mod/primitives/pkg/math"
"github.com/berachain/beacon-kit/mod/primitives/pkg/transition"
"github.com/sourcegraph/conc/iter"
)
Expand All @@ -36,8 +37,18 @@ func (sp *StateProcessor[
return nil, err
}

// filter out validators whose effective balance is not sufficient to validate
activeVals := make([]ValidatorT, 0, len(vals))
for _, val := range vals {
if val.GetEffectiveBalance() > math.U64(sp.cs.EjectionBalance()) {
activeVals = append(activeVals, val)
}
}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🧹 Nitpick (assertive)

Consider optimizing the validator filtering implementation.

While the filtering logic is correct, consider these improvements for better efficiency:

  1. Pre-allocate with a more precise capacity based on active validator ratio
  2. Use slice filtering patterns for better readability
-	activeVals := make([]ValidatorT, 0, len(vals))
-	for _, val := range vals {
-		if val.GetEffectiveBalance() > math.U64(sp.cs.EjectionBalance()) {
-			activeVals = append(activeVals, val)
-		}
-	}
+	// Estimate capacity based on typical active/total ratio (e.g., 80%)
+	estimatedCap := (len(vals) * 4) / 5
+	activeVals := make([]ValidatorT, 0, estimatedCap)
+	
+	// Use filter pattern for better readability
+	for i := range vals {
+		if vals[i].GetEffectiveBalance() > math.U64(sp.cs.EjectionBalance()) {
+			activeVals = append(activeVals, vals[i])
+		}
+	}
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
// filter out validators whose effective balance is not sufficient to validate
activeVals := make([]ValidatorT, 0, len(vals))
for _, val := range vals {
if val.GetEffectiveBalance() > math.U64(sp.cs.EjectionBalance()) {
activeVals = append(activeVals, val)
}
}
// filter out validators whose effective balance is not sufficient to validate
// Estimate capacity based on typical active/total ratio (e.g., 80%)
estimatedCap := (len(vals) * 4) / 5
activeVals := make([]ValidatorT, 0, estimatedCap)
// Use filter pattern for better readability
for i := range vals {
if vals[i].GetEffectiveBalance() > math.U64(sp.cs.EjectionBalance()) {
activeVals = append(activeVals, vals[i])
}
}


// TODO: a more efficient handling would be to only send back to consensus
// updated validators (including evicted ones), rather than the full list
return iter.MapErr(
vals,
activeVals,
func(val *ValidatorT) (*transition.ValidatorUpdate, error) {
v := (*val)
return &transition.ValidatorUpdate{
Expand Down
Loading
Loading