package beacon import ( "context" "testing" "time" chainMock "github.com/prysmaticlabs/prysm/v4/beacon-chain/blockchain/testing" dbTest "github.com/prysmaticlabs/prysm/v4/beacon-chain/db/testing" "github.com/prysmaticlabs/prysm/v4/beacon-chain/rpc/testutil" "github.com/prysmaticlabs/prysm/v4/config/params" "github.com/prysmaticlabs/prysm/v4/consensus-types/primitives" "github.com/prysmaticlabs/prysm/v4/encoding/bytesutil" eth "github.com/prysmaticlabs/prysm/v4/proto/eth/v1" eth2 "github.com/prysmaticlabs/prysm/v4/proto/eth/v2" ethpb "github.com/prysmaticlabs/prysm/v4/proto/prysm/v1alpha1" "github.com/prysmaticlabs/prysm/v4/testing/assert" "github.com/prysmaticlabs/prysm/v4/testing/require" "github.com/prysmaticlabs/prysm/v4/testing/util" "google.golang.org/protobuf/types/known/emptypb" ) func TestGetGenesis(t *testing.T) { ctx := context.Background() params.SetupTestConfigCleanup(t) config := params.BeaconConfig().Copy() config.GenesisForkVersion = []byte("genesis") params.OverrideBeaconConfig(config) genesis := time.Date(2021, 1, 1, 0, 0, 0, 0, time.UTC) validatorsRoot := [32]byte{1, 2, 3, 4, 5, 6} t.Run("OK", func(t *testing.T) { chainService := &chainMock.ChainService{ Genesis: genesis, ValidatorsRoot: validatorsRoot, } s := Server{ GenesisTimeFetcher: chainService, ChainInfoFetcher: chainService, } resp, err := s.GetGenesis(ctx, &emptypb.Empty{}) require.NoError(t, err) assert.Equal(t, genesis.Unix(), resp.Data.GenesisTime.Seconds) assert.Equal(t, int32(0), resp.Data.GenesisTime.Nanos) assert.DeepEqual(t, validatorsRoot[:], resp.Data.GenesisValidatorsRoot) assert.DeepEqual(t, []byte("genesis"), resp.Data.GenesisForkVersion) }) t.Run("No genesis time", func(t *testing.T) { chainService := &chainMock.ChainService{ Genesis: time.Time{}, ValidatorsRoot: validatorsRoot, } s := Server{ GenesisTimeFetcher: chainService, ChainInfoFetcher: chainService, } _, err := s.GetGenesis(ctx, &emptypb.Empty{}) assert.ErrorContains(t, "Chain genesis info is not yet known", err) }) t.Run("No genesis validators root", func(t *testing.T) { chainService := &chainMock.ChainService{ Genesis: genesis, ValidatorsRoot: [32]byte{}, } s := Server{ GenesisTimeFetcher: chainService, ChainInfoFetcher: chainService, } _, err := s.GetGenesis(ctx, &emptypb.Empty{}) assert.ErrorContains(t, "Chain genesis info is not yet known", err) }) } func TestGetStateRoot(t *testing.T) { ctx := context.Background() fakeState, err := util.NewBeaconState() require.NoError(t, err) stateRoot, err := fakeState.HashTreeRoot(ctx) require.NoError(t, err) db := dbTest.SetupDB(t) chainService := &chainMock.ChainService{} server := &Server{ StateFetcher: &testutil.MockFetcher{ BeaconStateRoot: stateRoot[:], BeaconState: fakeState, }, HeadFetcher: chainService, OptimisticModeFetcher: chainService, FinalizationFetcher: chainService, BeaconDB: db, } resp, err := server.GetStateRoot(context.Background(), ð.StateRequest{ StateId: []byte("head"), }) require.NoError(t, err) assert.NotNil(t, resp) assert.DeepEqual(t, stateRoot[:], resp.Data.Root) t.Run("execution optimistic", func(t *testing.T) { parentRoot := [32]byte{'a'} blk := util.NewBeaconBlock() blk.Block.ParentRoot = parentRoot[:] root, err := blk.Block.HashTreeRoot() require.NoError(t, err) util.SaveBlock(t, ctx, db, blk) require.NoError(t, db.SaveGenesisBlockRoot(ctx, root)) chainService := &chainMock.ChainService{Optimistic: true} server := &Server{ StateFetcher: &testutil.MockFetcher{ BeaconStateRoot: stateRoot[:], BeaconState: fakeState, }, HeadFetcher: chainService, OptimisticModeFetcher: chainService, FinalizationFetcher: chainService, BeaconDB: db, } resp, err := server.GetStateRoot(context.Background(), ð.StateRequest{ StateId: []byte("head"), }) require.NoError(t, err) assert.NotNil(t, resp) assert.DeepEqual(t, true, resp.ExecutionOptimistic) }) t.Run("finalized", func(t *testing.T) { parentRoot := [32]byte{'a'} blk := util.NewBeaconBlock() blk.Block.ParentRoot = parentRoot[:] root, err := blk.Block.HashTreeRoot() require.NoError(t, err) util.SaveBlock(t, ctx, db, blk) require.NoError(t, db.SaveGenesisBlockRoot(ctx, root)) headerRoot, err := fakeState.LatestBlockHeader().HashTreeRoot() require.NoError(t, err) chainService := &chainMock.ChainService{ FinalizedRoots: map[[32]byte]bool{ headerRoot: true, }, } server := &Server{ StateFetcher: &testutil.MockFetcher{ BeaconStateRoot: stateRoot[:], BeaconState: fakeState, }, HeadFetcher: chainService, OptimisticModeFetcher: chainService, FinalizationFetcher: chainService, BeaconDB: db, } resp, err := server.GetStateRoot(context.Background(), ð.StateRequest{ StateId: []byte("head"), }) require.NoError(t, err) assert.NotNil(t, resp) assert.DeepEqual(t, true, resp.Finalized) }) } func TestGetStateFork(t *testing.T) { ctx := context.Background() fillFork := func(state *ethpb.BeaconState) error { state.Fork = ðpb.Fork{ PreviousVersion: []byte("prev"), CurrentVersion: []byte("curr"), Epoch: 123, } return nil } fakeState, err := util.NewBeaconState(fillFork) require.NoError(t, err) db := dbTest.SetupDB(t) chainService := &chainMock.ChainService{} server := &Server{ StateFetcher: &testutil.MockFetcher{ BeaconState: fakeState, }, HeadFetcher: chainService, OptimisticModeFetcher: chainService, FinalizationFetcher: chainService, BeaconDB: db, } resp, err := server.GetStateFork(ctx, ð.StateRequest{ StateId: []byte("head"), }) require.NoError(t, err) assert.NotNil(t, resp) expectedFork := fakeState.Fork() assert.Equal(t, expectedFork.Epoch, resp.Data.Epoch) assert.DeepEqual(t, expectedFork.CurrentVersion, resp.Data.CurrentVersion) assert.DeepEqual(t, expectedFork.PreviousVersion, resp.Data.PreviousVersion) t.Run("execution optimistic", func(t *testing.T) { parentRoot := [32]byte{'a'} blk := util.NewBeaconBlock() blk.Block.ParentRoot = parentRoot[:] root, err := blk.Block.HashTreeRoot() require.NoError(t, err) util.SaveBlock(t, ctx, db, blk) require.NoError(t, db.SaveGenesisBlockRoot(ctx, root)) chainService := &chainMock.ChainService{Optimistic: true} server := &Server{ StateFetcher: &testutil.MockFetcher{ BeaconState: fakeState, }, HeadFetcher: chainService, OptimisticModeFetcher: chainService, FinalizationFetcher: chainService, BeaconDB: db, } resp, err := server.GetStateFork(context.Background(), ð.StateRequest{ StateId: []byte("head"), }) require.NoError(t, err) assert.NotNil(t, resp) assert.DeepEqual(t, true, resp.ExecutionOptimistic) }) t.Run("finalized", func(t *testing.T) { parentRoot := [32]byte{'a'} blk := util.NewBeaconBlock() blk.Block.ParentRoot = parentRoot[:] root, err := blk.Block.HashTreeRoot() require.NoError(t, err) util.SaveBlock(t, ctx, db, blk) require.NoError(t, db.SaveGenesisBlockRoot(ctx, root)) headerRoot, err := fakeState.LatestBlockHeader().HashTreeRoot() require.NoError(t, err) chainService := &chainMock.ChainService{ FinalizedRoots: map[[32]byte]bool{ headerRoot: true, }, } server := &Server{ StateFetcher: &testutil.MockFetcher{ BeaconState: fakeState, }, HeadFetcher: chainService, OptimisticModeFetcher: chainService, FinalizationFetcher: chainService, BeaconDB: db, } resp, err := server.GetStateFork(context.Background(), ð.StateRequest{ StateId: []byte("head"), }) require.NoError(t, err) assert.NotNil(t, resp) assert.DeepEqual(t, true, resp.Finalized) }) } func TestGetFinalityCheckpoints(t *testing.T) { ctx := context.Background() fillCheckpoints := func(state *ethpb.BeaconState) error { state.PreviousJustifiedCheckpoint = ðpb.Checkpoint{ Root: bytesutil.PadTo([]byte("previous"), 32), Epoch: 113, } state.CurrentJustifiedCheckpoint = ðpb.Checkpoint{ Root: bytesutil.PadTo([]byte("current"), 32), Epoch: 123, } state.FinalizedCheckpoint = ðpb.Checkpoint{ Root: bytesutil.PadTo([]byte("finalized"), 32), Epoch: 103, } return nil } fakeState, err := util.NewBeaconState(fillCheckpoints) require.NoError(t, err) db := dbTest.SetupDB(t) chainService := &chainMock.ChainService{} server := &Server{ StateFetcher: &testutil.MockFetcher{ BeaconState: fakeState, }, HeadFetcher: chainService, OptimisticModeFetcher: chainService, FinalizationFetcher: chainService, BeaconDB: db, } resp, err := server.GetFinalityCheckpoints(ctx, ð.StateRequest{ StateId: []byte("head"), }) require.NoError(t, err) assert.NotNil(t, resp) assert.Equal(t, fakeState.FinalizedCheckpoint().Epoch, resp.Data.Finalized.Epoch) assert.DeepEqual(t, fakeState.FinalizedCheckpoint().Root, resp.Data.Finalized.Root) assert.Equal(t, fakeState.CurrentJustifiedCheckpoint().Epoch, resp.Data.CurrentJustified.Epoch) assert.DeepEqual(t, fakeState.CurrentJustifiedCheckpoint().Root, resp.Data.CurrentJustified.Root) assert.Equal(t, fakeState.PreviousJustifiedCheckpoint().Epoch, resp.Data.PreviousJustified.Epoch) assert.DeepEqual(t, fakeState.PreviousJustifiedCheckpoint().Root, resp.Data.PreviousJustified.Root) t.Run("execution optimistic", func(t *testing.T) { parentRoot := [32]byte{'a'} blk := util.NewBeaconBlock() blk.Block.ParentRoot = parentRoot[:] root, err := blk.Block.HashTreeRoot() require.NoError(t, err) util.SaveBlock(t, ctx, db, blk) require.NoError(t, db.SaveGenesisBlockRoot(ctx, root)) chainService := &chainMock.ChainService{Optimistic: true} server := &Server{ StateFetcher: &testutil.MockFetcher{ BeaconState: fakeState, }, HeadFetcher: chainService, OptimisticModeFetcher: chainService, FinalizationFetcher: chainService, BeaconDB: db, } resp, err := server.GetFinalityCheckpoints(context.Background(), ð.StateRequest{ StateId: []byte("head"), }) require.NoError(t, err) assert.NotNil(t, resp) assert.DeepEqual(t, true, resp.ExecutionOptimistic) }) t.Run("finalized", func(t *testing.T) { parentRoot := [32]byte{'a'} blk := util.NewBeaconBlock() blk.Block.ParentRoot = parentRoot[:] root, err := blk.Block.HashTreeRoot() require.NoError(t, err) util.SaveBlock(t, ctx, db, blk) require.NoError(t, db.SaveGenesisBlockRoot(ctx, root)) headerRoot, err := fakeState.LatestBlockHeader().HashTreeRoot() require.NoError(t, err) chainService := &chainMock.ChainService{ FinalizedRoots: map[[32]byte]bool{ headerRoot: true, }, } server := &Server{ StateFetcher: &testutil.MockFetcher{ BeaconState: fakeState, }, HeadFetcher: chainService, OptimisticModeFetcher: chainService, FinalizationFetcher: chainService, BeaconDB: db, } resp, err := server.GetFinalityCheckpoints(context.Background(), ð.StateRequest{ StateId: []byte("head"), }) require.NoError(t, err) assert.NotNil(t, resp) assert.DeepEqual(t, true, resp.Finalized) }) } func TestGetRandao(t *testing.T) { mixCurrent := bytesutil.PadTo([]byte("current"), 32) mixOld := bytesutil.PadTo([]byte("old"), 32) epochCurrent := primitives.Epoch(100000) epochOld := 100000 - params.BeaconConfig().EpochsPerHistoricalVector + 1 ctx := context.Background() st, err := util.NewBeaconState() require.NoError(t, err) // Set slot to epoch 100000 require.NoError(t, st.SetSlot(params.BeaconConfig().SlotsPerEpoch*100000)) require.NoError(t, st.UpdateRandaoMixesAtIndex(uint64(epochCurrent%params.BeaconConfig().EpochsPerHistoricalVector), mixCurrent)) require.NoError(t, st.UpdateRandaoMixesAtIndex(uint64(epochOld%params.BeaconConfig().EpochsPerHistoricalVector), mixOld)) headEpoch := primitives.Epoch(1) headSt, err := util.NewBeaconState() require.NoError(t, err) require.NoError(t, headSt.SetSlot(params.BeaconConfig().SlotsPerEpoch)) headRandao := bytesutil.PadTo([]byte("head"), 32) require.NoError(t, headSt.UpdateRandaoMixesAtIndex(uint64(headEpoch), headRandao)) db := dbTest.SetupDB(t) chainService := &chainMock.ChainService{} server := &Server{ StateFetcher: &testutil.MockFetcher{ BeaconState: st, }, HeadFetcher: chainService, OptimisticModeFetcher: chainService, FinalizationFetcher: chainService, BeaconDB: db, } t.Run("no epoch requested", func(t *testing.T) { resp, err := server.GetRandao(ctx, ð2.RandaoRequest{StateId: []byte("head")}) require.NoError(t, err) assert.DeepEqual(t, mixCurrent, resp.Data.Randao) }) t.Run("current epoch requested", func(t *testing.T) { resp, err := server.GetRandao(ctx, ð2.RandaoRequest{StateId: []byte("head"), Epoch: &epochCurrent}) require.NoError(t, err) assert.DeepEqual(t, mixCurrent, resp.Data.Randao) }) t.Run("old epoch requested", func(t *testing.T) { resp, err := server.GetRandao(ctx, ð2.RandaoRequest{StateId: []byte("head"), Epoch: &epochOld}) require.NoError(t, err) assert.DeepEqual(t, mixOld, resp.Data.Randao) }) t.Run("head state below `EpochsPerHistoricalVector`", func(t *testing.T) { server.StateFetcher = &testutil.MockFetcher{ BeaconState: headSt, } resp, err := server.GetRandao(ctx, ð2.RandaoRequest{StateId: []byte("head")}) require.NoError(t, err) assert.DeepEqual(t, headRandao, resp.Data.Randao) }) t.Run("epoch too old", func(t *testing.T) { epochTooOld := primitives.Epoch(100000 - st.RandaoMixesLength()) _, err := server.GetRandao(ctx, ð2.RandaoRequest{StateId: make([]byte, 0), Epoch: &epochTooOld}) require.ErrorContains(t, "Epoch is out of range for the randao mixes of the state", err) }) t.Run("epoch in the future", func(t *testing.T) { futureEpoch := primitives.Epoch(100000 + 1) _, err := server.GetRandao(ctx, ð2.RandaoRequest{StateId: make([]byte, 0), Epoch: &futureEpoch}) require.ErrorContains(t, "Epoch is out of range for the randao mixes of the state", err) }) t.Run("execution optimistic", func(t *testing.T) { parentRoot := [32]byte{'a'} blk := util.NewBeaconBlock() blk.Block.ParentRoot = parentRoot[:] root, err := blk.Block.HashTreeRoot() require.NoError(t, err) util.SaveBlock(t, ctx, db, blk) require.NoError(t, db.SaveGenesisBlockRoot(ctx, root)) chainService := &chainMock.ChainService{Optimistic: true} server := &Server{ StateFetcher: &testutil.MockFetcher{ BeaconState: st, }, HeadFetcher: chainService, OptimisticModeFetcher: chainService, FinalizationFetcher: chainService, BeaconDB: db, } resp, err := server.GetRandao(context.Background(), ð2.RandaoRequest{ StateId: []byte("head"), }) require.NoError(t, err) assert.NotNil(t, resp) assert.DeepEqual(t, true, resp.ExecutionOptimistic) }) t.Run("finalized", func(t *testing.T) { parentRoot := [32]byte{'a'} blk := util.NewBeaconBlock() blk.Block.ParentRoot = parentRoot[:] root, err := blk.Block.HashTreeRoot() require.NoError(t, err) util.SaveBlock(t, ctx, db, blk) require.NoError(t, db.SaveGenesisBlockRoot(ctx, root)) headerRoot, err := headSt.LatestBlockHeader().HashTreeRoot() require.NoError(t, err) chainService := &chainMock.ChainService{ FinalizedRoots: map[[32]byte]bool{ headerRoot: true, }, } server := &Server{ StateFetcher: &testutil.MockFetcher{ BeaconState: st, }, HeadFetcher: chainService, OptimisticModeFetcher: chainService, FinalizationFetcher: chainService, BeaconDB: db, } resp, err := server.GetRandao(context.Background(), ð2.RandaoRequest{ StateId: []byte("head"), }) require.NoError(t, err) assert.NotNil(t, resp) assert.DeepEqual(t, true, resp.Finalized) }) }