package node import ( "context" "fmt" "testing" "time" "github.com/ethereum/go-ethereum/common" "github.com/ethereum/go-ethereum/crypto" "github.com/ethereum/go-ethereum/p2p/enode" mock "github.com/prysmaticlabs/prysm/beacon-chain/blockchain/testing" dbutil "github.com/prysmaticlabs/prysm/beacon-chain/db/testing" "github.com/prysmaticlabs/prysm/beacon-chain/p2p" mockP2p "github.com/prysmaticlabs/prysm/beacon-chain/p2p/testing" "github.com/prysmaticlabs/prysm/beacon-chain/rpc/testutil" mockSync "github.com/prysmaticlabs/prysm/beacon-chain/sync/initial-sync/testing" "github.com/prysmaticlabs/prysm/encoding/bytesutil" ethpb "github.com/prysmaticlabs/prysm/proto/prysm/v1alpha1" "github.com/prysmaticlabs/prysm/runtime/version" "github.com/prysmaticlabs/prysm/testing/assert" "github.com/prysmaticlabs/prysm/testing/require" "github.com/prysmaticlabs/prysm/testing/util" "google.golang.org/grpc" "google.golang.org/grpc/reflection" "google.golang.org/protobuf/types/known/emptypb" "google.golang.org/protobuf/types/known/timestamppb" ) func TestNodeServer_GetSyncStatus(t *testing.T) { mSync := &mockSync.Sync{IsSyncing: false} ns := &Server{ SyncChecker: mSync, } res, err := ns.GetSyncStatus(context.Background(), &emptypb.Empty{}) require.NoError(t, err) assert.Equal(t, false, res.Syncing) ns.SyncChecker = &mockSync.Sync{IsSyncing: true} res, err = ns.GetSyncStatus(context.Background(), &emptypb.Empty{}) require.NoError(t, err) assert.Equal(t, true, res.Syncing) } func TestNodeServer_GetGenesis(t *testing.T) { db := dbutil.SetupDB(t) ctx := context.Background() addr := common.Address{1, 2, 3} require.NoError(t, db.SaveDepositContractAddress(ctx, addr)) st, err := util.NewBeaconState() require.NoError(t, err) genValRoot := bytesutil.ToBytes32([]byte("I am root")) ns := &Server{ BeaconDB: db, GenesisTimeFetcher: &mock.ChainService{}, GenesisFetcher: &mock.ChainService{ State: st, ValidatorsRoot: genValRoot, }, } res, err := ns.GetGenesis(context.Background(), &emptypb.Empty{}) require.NoError(t, err) assert.DeepEqual(t, addr.Bytes(), res.DepositContractAddress) pUnix := timestamppb.New(time.Unix(0, 0)) assert.Equal(t, res.GenesisTime.Seconds, pUnix.Seconds) assert.DeepEqual(t, genValRoot[:], res.GenesisValidatorsRoot) ns.GenesisTimeFetcher = &mock.ChainService{Genesis: time.Unix(10, 0)} res, err = ns.GetGenesis(context.Background(), &emptypb.Empty{}) require.NoError(t, err) pUnix = timestamppb.New(time.Unix(10, 0)) assert.Equal(t, res.GenesisTime.Seconds, pUnix.Seconds) } func TestNodeServer_GetVersion(t *testing.T) { v := version.Version() ns := &Server{} res, err := ns.GetVersion(context.Background(), &emptypb.Empty{}) require.NoError(t, err) assert.Equal(t, v, res.Version) } func TestNodeServer_GetImplementedServices(t *testing.T) { server := grpc.NewServer() ns := &Server{ Server: server, } ethpb.RegisterNodeServer(server, ns) reflection.Register(server) res, err := ns.ListImplementedServices(context.Background(), &emptypb.Empty{}) require.NoError(t, err) // We verify the services include the node service + the registered reflection service. assert.Equal(t, 2, len(res.Services)) } func TestNodeServer_GetHost(t *testing.T) { server := grpc.NewServer() peersProvider := &mockP2p.MockPeersProvider{} mP2P := mockP2p.NewTestP2P(t) key, err := crypto.GenerateKey() require.NoError(t, err) db, err := enode.OpenDB("") require.NoError(t, err) lNode := enode.NewLocalNode(db, key) record := lNode.Node().Record() stringENR, err := p2p.SerializeENR(record) require.NoError(t, err) ns := &Server{ PeerManager: &mockP2p.MockPeerManager{BHost: mP2P.BHost, Enr: record, PID: mP2P.BHost.ID()}, PeersFetcher: peersProvider, } ethpb.RegisterNodeServer(server, ns) reflection.Register(server) h, err := ns.GetHost(context.Background(), &emptypb.Empty{}) require.NoError(t, err) assert.Equal(t, mP2P.PeerID().String(), h.PeerId) assert.Equal(t, stringENR, h.Enr) } func TestNodeServer_GetPeer(t *testing.T) { server := grpc.NewServer() peersProvider := &mockP2p.MockPeersProvider{} ns := &Server{ PeersFetcher: peersProvider, } ethpb.RegisterNodeServer(server, ns) reflection.Register(server) firstPeer := peersProvider.Peers().All()[0] res, err := ns.GetPeer(context.Background(), ðpb.PeerRequest{PeerId: firstPeer.String()}) require.NoError(t, err) assert.Equal(t, firstPeer.String(), res.PeerId, "Unexpected peer ID") assert.Equal(t, int(ethpb.PeerDirection_INBOUND), int(res.Direction), "Expected 1st peer to be an inbound connection") assert.Equal(t, ethpb.ConnectionState_CONNECTED, res.ConnectionState, "Expected peer to be connected") } func TestNodeServer_ListPeers(t *testing.T) { server := grpc.NewServer() peersProvider := &mockP2p.MockPeersProvider{} ns := &Server{ PeersFetcher: peersProvider, } ethpb.RegisterNodeServer(server, ns) reflection.Register(server) res, err := ns.ListPeers(context.Background(), &emptypb.Empty{}) require.NoError(t, err) assert.Equal(t, 2, len(res.Peers)) assert.Equal(t, int(ethpb.PeerDirection_INBOUND), int(res.Peers[0].Direction)) assert.Equal(t, ethpb.PeerDirection_OUTBOUND, res.Peers[1].Direction) } func TestNodeServer_GetETH1ConnectionStatus(t *testing.T) { server := grpc.NewServer() eps := []string{"foo", "bar"} errs := []error{fmt.Errorf("error 1"), fmt.Errorf("error 2"), nil} errStrs := []string{"error 1", "error 2", ""} mockFetcher := &testutil.MockPOWChainInfoFetcher{ CurrEndpoint: eps[0], CurrError: errs[0], Endpoints: eps, Errors: errs, } ns := &Server{ POWChainInfoFetcher: mockFetcher, } ethpb.RegisterNodeServer(server, ns) reflection.Register(server) res, err := ns.GetETH1ConnectionStatus(context.Background(), &emptypb.Empty{}) require.NoError(t, err) assert.Equal(t, eps[0], res.CurrentAddress) assert.Equal(t, errStrs[0], res.CurrentConnectionError) assert.DeepSSZEqual(t, eps, res.Addresses) assert.DeepSSZEqual(t, errStrs, res.ConnectionErrors) }