From 901a9489f641cb3196fa5d462d03f5bb2c4a9227 Mon Sep 17 00:00:00 2001 From: Tarun Date: Wed, 26 Oct 2022 18:43:52 +0200 Subject: [PATCH 1/4] Fix file not compiling properly --- pkg/utils/validator_indexes.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pkg/utils/validator_indexes.go b/pkg/utils/validator_indexes.go index 2ca6b8f9..1408bf86 100644 --- a/pkg/utils/validator_indexes.go +++ b/pkg/utils/validator_indexes.go @@ -16,7 +16,7 @@ func GetValIndexesFromJson(filePath string) ([]uint64, error) { err = json.Unmarshal(fbytes, &validatorIndex) if err != nil { - log.Errorf("Error unmarshalling val list", err.Error()) + log.Errorf("Error unmarshalling val list: %s", err.Error()) } log.Infof("Readed %d validators", len(validatorIndex)) From 5dcd1ed67bb863d395030ce08a0d1934ce9a02d4 Mon Sep 17 00:00:00 2001 From: Tarun Date: Wed, 26 Oct 2022 18:44:43 +0200 Subject: [PATCH 2/4] Add state tests --- pkg/fork_metrics/fork_state/state_test.go | 83 +++++++++++++++++++++++ 1 file changed, 83 insertions(+) create mode 100644 pkg/fork_metrics/fork_state/state_test.go diff --git a/pkg/fork_metrics/fork_state/state_test.go b/pkg/fork_metrics/fork_state/state_test.go new file mode 100644 index 00000000..886fbd0d --- /dev/null +++ b/pkg/fork_metrics/fork_state/state_test.go @@ -0,0 +1,83 @@ +package fork_state + +import ( + "testing" + + "github.com/attestantio/go-eth2-client/spec/altair" + "github.com/attestantio/go-eth2-client/spec/phase0" + "github.com/stretchr/testify/require" +) + +func TestState(t *testing.T) { + + balancesArray := make([]uint64, 0) + balancesArray = append(balancesArray, 34000000000, 31200000000) + validator1 := phase0.Validator{ + EffectiveBalance: 32000000000, + ActivationEpoch: 0, + ExitEpoch: 10000000000, + } + + validator2 := phase0.Validator{ + EffectiveBalance: 31000000000, + ActivationEpoch: 0, + ExitEpoch: 10000000000, + } + + validatorArray := make([]*phase0.Validator, 0) + validatorArray = append(validatorArray, &validator1, &validator2) + + state := ForkStateContentBase{ + Validators: validatorArray, + Balances: balancesArray, + BlockRoots: make([][]byte, 0), + } + state.Setup() + + require.Equal(t, state.TotalActiveBalance, uint64(validator1.EffectiveBalance+validator2.EffectiveBalance)) + + state.Validators = append(state.Validators, &validator1, &validator2) + + state = ForkStateContentBase{ + Validators: validatorArray, + Balances: balancesArray, + BlockRoots: make([][]byte, 0), + } + state.Setup() + attestations := make([]altair.ParticipationFlags, 0) + attestations = append(attestations, altair.ParticipationFlags(7)) + + ProcessAttestations(&state, attestations) + + require.Equal(t, state.AttestingBalance[0], uint64(validator1.EffectiveBalance)) + require.Equal(t, state.AttestingVals[0], true) + require.Equal(t, state.AttestingVals[1], false) + + attestations = append(attestations, altair.ParticipationFlags(7)) + state = ForkStateContentBase{ + Validators: validatorArray, + Balances: balancesArray, + BlockRoots: make([][]byte, 0), + } + state.Setup() + ProcessAttestations(&state, attestations) + + require.Equal(t, state.AttestingBalance[0], uint64(validator1.EffectiveBalance+validator2.EffectiveBalance)) + require.Equal(t, state.AttestingVals[0], true) + require.Equal(t, state.AttestingVals[1], true) + + state = ForkStateContentBase{ + Validators: validatorArray, + Balances: balancesArray, + BlockRoots: make([][]byte, 0), + } + state.Setup() + attestations[1] = altair.ParticipationFlags(6) + ProcessAttestations(&state, attestations) + + // no source attesting + require.Equal(t, state.AttestingBalance[0], uint64(validator1.EffectiveBalance)) + require.Equal(t, state.AttestingVals[0], true) + require.Equal(t, state.AttestingVals[1], true) + +} From a9b87706b4c9685ae41f5e0e0703d7b5ff5d41dd Mon Sep 17 00:00:00 2001 From: Tarun Date: Thu, 27 Oct 2022 18:29:08 +0200 Subject: [PATCH 3/4] Add Rewards Tests --- pkg/fork_metrics/rewards_test.go | 154 +++++++++++++++++++++++++++++++ 1 file changed, 154 insertions(+) create mode 100644 pkg/fork_metrics/rewards_test.go diff --git a/pkg/fork_metrics/rewards_test.go b/pkg/fork_metrics/rewards_test.go new file mode 100644 index 00000000..a2c29f7a --- /dev/null +++ b/pkg/fork_metrics/rewards_test.go @@ -0,0 +1,154 @@ +package fork_metrics + +import ( + "testing" + + "github.com/attestantio/go-eth2-client/spec/altair" + "github.com/attestantio/go-eth2-client/spec/phase0" + "github.com/cortze/eth2-state-analyzer/pkg/fork_metrics/fork_state" + "github.com/stretchr/testify/require" +) + +func TestMaxAttestationReward(t *testing.T) { + + balancesArray := make([]uint64, 0) + balancesArray = append(balancesArray, 34000000000, 31200000000) + validator1 := phase0.Validator{ + EffectiveBalance: 32000000000, + ActivationEpoch: 0, + ExitEpoch: 10000000000, + } + + validator2 := phase0.Validator{ + EffectiveBalance: 31000000000, + ActivationEpoch: 0, + ExitEpoch: 10000000000, + } + + validatorArray := make([]*phase0.Validator, 0) + validatorArray = append(validatorArray, &validator1, &validator2) + + state := fork_state.ForkStateContentBase{ + Validators: validatorArray, + Balances: balancesArray, + BlockRoots: make([][]byte, 0), + Epoch: 10, + } + state.Setup() + + balancesArray1 := make([]uint64, 0) + balancesArray1 = append(balancesArray1, 34000000000, 31200000000) + validator1 = phase0.Validator{ + EffectiveBalance: 32000000000, + ActivationEpoch: 0, + ExitEpoch: 10000000000, + } + + validator2 = phase0.Validator{ + EffectiveBalance: 31000000000, + ActivationEpoch: 0, + ExitEpoch: 10000000000, + } + + validatorArray1 := make([]*phase0.Validator, 0) + validatorArray1 = append(validatorArray1, &validator1, &validator2) + + statePrev := fork_state.ForkStateContentBase{ + Validators: validatorArray1, + Balances: balancesArray1, + BlockRoots: make([][]byte, 0), + Epoch: 9, + } + + attestations := make([]altair.ParticipationFlags, 0) + attestations = append(attestations, altair.ParticipationFlags(6)) + statePrev.Setup() + fork_state.ProcessAttestations(&state, attestations) + + stateNext := statePrev + stateNext.Epoch = 11 + rewardsObj := NewAltairMetrics( + statePrev, + state, + stateNext) + + require.Equal(t, + rewardsObj.GetBaseRewardPerInc(rewardsObj.CurrentState.TotalActiveBalance), + uint64(254982)) + + require.Equal(t, + rewardsObj.GetBaseReward(1, uint64(validator1.EffectiveBalance), rewardsObj.CurrentState.TotalActiveBalance), + uint64(254982*32)) + + require.Equal(t, + rewardsObj.GetMaxAttestationReward(1), + uint64(2509346)) + + require.Equal(t, + rewardsObj.GetMaxAttestationReward(0), + uint64(2590292)) + +} + +func TestMaxSyncCommitteeReward(t *testing.T) { + + // create state + balancesArray := make([]uint64, 0) + balancesArray = append(balancesArray, 34000000000, 31200000000) + validator1Pubkey := "0x8b9cfaf7480d7bb848cc2017b9770bef00f8d9e761bea9ea06a0534c449a98f50b587db18a93c4da8ae805476edee55a" + validator1 := phase0.Validator{ + EffectiveBalance: 32000000000, + ActivationEpoch: 0, + ExitEpoch: 10000000000, + PublicKey: phase0.BLSPubKey{}, + } + + copy(validator1.PublicKey[:], validator1Pubkey) + + validator2Pubkey := "0x9b9cfaf7480d7bb848cc2017b9770bef00f8d9e761bea9ea06a0534c449a98f50b587db18a93c4da8ae805476edee55a" + validator2 := phase0.Validator{ + EffectiveBalance: 31000000000, + ActivationEpoch: 0, + ExitEpoch: 10000000000, + PublicKey: phase0.BLSPubKey{}, + } + copy(validator1.PublicKey[:], validator2Pubkey) + + validatorArray := make([]*phase0.Validator, 0) + validatorArray = append(validatorArray, &validator1, &validator2) + syncCommittee := altair.SyncCommittee{ + Pubkeys: make([]phase0.BLSPubKey, 0), + } + syncCommittee.Pubkeys = append(syncCommittee.Pubkeys, validator1.PublicKey) // validator 1 is in sync committee + + // state creation + stateNext := fork_state.ForkStateContentBase{ + Validators: validatorArray, + Balances: balancesArray, + BlockRoots: make([][]byte, 0), + Epoch: 11, + SyncCommittee: syncCommittee, + } + stateNext.Setup() + + // create fork metrics + rewardsObj := NewAltairMetrics( + stateNext, + fork_state.ForkStateContentBase{}, + fork_state.ForkStateContentBase{}) + + require.Equal(t, + rewardsObj.GetMaxSyncComReward(0), + uint64(30*32)) + + require.Equal(t, + rewardsObj.GetMaxSyncComReward(1), + uint64(0*32)) + + rewardsObj.NextState.MissedBlocks = append(rewardsObj.NextState.MissedBlocks, 1) + + require.Equal(t, + rewardsObj.GetMaxSyncComReward(0), + uint64(30*31)) + +} From e1c9fbf21853c36178bb2a19412bf22f2dcac711 Mon Sep 17 00:00:00 2001 From: Tarun Date: Thu, 27 Oct 2022 18:47:39 +0200 Subject: [PATCH 4/4] Improved tests to check specification formulas --- pkg/fork_metrics/rewards_test.go | 61 ++++++++++++++++++++++++++++---- 1 file changed, 54 insertions(+), 7 deletions(-) diff --git a/pkg/fork_metrics/rewards_test.go b/pkg/fork_metrics/rewards_test.go index a2c29f7a..453ee049 100644 --- a/pkg/fork_metrics/rewards_test.go +++ b/pkg/fork_metrics/rewards_test.go @@ -1,6 +1,7 @@ package fork_metrics import ( + "math" "testing" "github.com/attestantio/go-eth2-client/spec/altair" @@ -72,18 +73,57 @@ func TestMaxAttestationReward(t *testing.T) { state, stateNext) + baseRewardPerInc := uint64(fork_state.EFFECTIVE_BALANCE_INCREMENT * fork_state.BASE_REWARD_FACTOR) + baseRewardPerInc = baseRewardPerInc / uint64(math.Sqrt(float64(rewardsObj.CurrentState.TotalActiveBalance))) require.Equal(t, rewardsObj.GetBaseRewardPerInc(rewardsObj.CurrentState.TotalActiveBalance), - uint64(254982)) + uint64(baseRewardPerInc)) require.Equal(t, - rewardsObj.GetBaseReward(1, uint64(validator1.EffectiveBalance), rewardsObj.CurrentState.TotalActiveBalance), - uint64(254982*32)) + rewardsObj.GetBaseReward(0, uint64(validator1.EffectiveBalance), rewardsObj.CurrentState.TotalActiveBalance), + uint64(baseRewardPerInc*32)) + + require.Equal(t, + rewardsObj.GetBaseReward(1, uint64(validator2.EffectiveBalance), rewardsObj.CurrentState.TotalActiveBalance), + uint64(baseRewardPerInc*31)) + + attReward := 0 + + // Source + reward := uint64(14) * baseRewardPerInc * 31 * uint64(rewardsObj.CurrentState.AttestingBalance[0]/1000000000) + reward = reward / (uint64(rewardsObj.CurrentState.TotalActiveBalance / 1000000000)) / 64 + attReward += int(reward) + + // Target + reward = uint64(26) * baseRewardPerInc * 31 * uint64(rewardsObj.CurrentState.AttestingBalance[1]/1000000000) + reward = reward / (uint64(rewardsObj.CurrentState.TotalActiveBalance / 1000000000)) / 64 + attReward += int(reward) + + // Head + reward = uint64(14) * baseRewardPerInc * 31 * uint64(rewardsObj.CurrentState.AttestingBalance[2]/1000000000) + reward = reward / (uint64(rewardsObj.CurrentState.TotalActiveBalance / 1000000000)) / 64 + attReward += int(reward) require.Equal(t, rewardsObj.GetMaxAttestationReward(1), - uint64(2509346)) + uint64(attReward)) + + attReward = 0 + + // Source + reward = uint64(14) * baseRewardPerInc * 32 * uint64(rewardsObj.CurrentState.AttestingBalance[0]/1000000000) + reward = reward / (uint64(rewardsObj.CurrentState.TotalActiveBalance / 1000000000)) / 64 + attReward += int(reward) + // Target + reward = uint64(26) * baseRewardPerInc * 32 * uint64(rewardsObj.CurrentState.AttestingBalance[1]/1000000000) + reward = reward / (uint64(rewardsObj.CurrentState.TotalActiveBalance / 1000000000)) / 64 + attReward += int(reward) + + // Head + reward = uint64(14) * baseRewardPerInc * 32 * uint64(rewardsObj.CurrentState.AttestingBalance[2]/1000000000) + reward = reward / (uint64(rewardsObj.CurrentState.TotalActiveBalance / 1000000000)) / 64 + attReward += int(reward) require.Equal(t, rewardsObj.GetMaxAttestationReward(0), uint64(2590292)) @@ -137,18 +177,25 @@ func TestMaxSyncCommitteeReward(t *testing.T) { fork_state.ForkStateContentBase{}, fork_state.ForkStateContentBase{}) + baseRewardPerInc := uint64(fork_state.EFFECTIVE_BALANCE_INCREMENT * fork_state.BASE_REWARD_FACTOR) + baseRewardPerInc = baseRewardPerInc / uint64(math.Sqrt(float64(rewardsObj.NextState.TotalActiveBalance))) + + participantReward := uint64(rewardsObj.NextState.TotalActiveBalance / 1000000000) + participantReward = participantReward * baseRewardPerInc + participantReward = participantReward * 2 / 64 / 32 + participantReward = participantReward / 512 require.Equal(t, rewardsObj.GetMaxSyncComReward(0), - uint64(30*32)) + uint64(participantReward*32)) require.Equal(t, rewardsObj.GetMaxSyncComReward(1), - uint64(0*32)) + uint64(0)) // not in sync committee rewardsObj.NextState.MissedBlocks = append(rewardsObj.NextState.MissedBlocks, 1) require.Equal(t, rewardsObj.GetMaxSyncComReward(0), - uint64(30*31)) + uint64(participantReward*31)) // one missed block }