mirror of https://github.com/grafana/grafana.git
349 lines
9.4 KiB
Go
349 lines
9.4 KiB
Go
package navtreeimpl
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"net/http"
|
|
"testing"
|
|
|
|
"github.com/stretchr/testify/mock"
|
|
"github.com/stretchr/testify/require"
|
|
|
|
"github.com/grafana/grafana/pkg/plugins"
|
|
"github.com/grafana/grafana/pkg/services/accesscontrol/actest"
|
|
contextmodel "github.com/grafana/grafana/pkg/services/contexthandler/model"
|
|
"github.com/grafana/grafana/pkg/services/dashboards"
|
|
"github.com/grafana/grafana/pkg/services/navtree"
|
|
"github.com/grafana/grafana/pkg/services/pluginsintegration/pluginstore"
|
|
"github.com/grafana/grafana/pkg/services/search/model"
|
|
"github.com/grafana/grafana/pkg/services/star"
|
|
"github.com/grafana/grafana/pkg/services/star/startest"
|
|
"github.com/grafana/grafana/pkg/services/user"
|
|
"github.com/grafana/grafana/pkg/setting"
|
|
"github.com/grafana/grafana/pkg/web"
|
|
)
|
|
|
|
func TestBuildStarredItemsNavLinks(t *testing.T) {
|
|
httpReq, _ := http.NewRequest(http.MethodGet, "", nil)
|
|
reqCtx := &contextmodel.ReqContext{
|
|
SignedInUser: &user.SignedInUser{
|
|
UserID: 1,
|
|
OrgID: 1,
|
|
},
|
|
Context: &web.Context{Req: httpReq},
|
|
}
|
|
|
|
t.Run("Should return empty list when there are no starred dashboards", func(t *testing.T) {
|
|
starService := startest.NewStarServiceFake()
|
|
starService.ExpectedUserStars = &star.GetUserStarsResult{
|
|
UserStars: map[string]bool{},
|
|
}
|
|
|
|
service := ServiceImpl{
|
|
starService: starService,
|
|
}
|
|
|
|
navLinks, err := service.buildStarredItemsNavLinks(reqCtx)
|
|
require.NoError(t, err)
|
|
require.Empty(t, navLinks)
|
|
})
|
|
|
|
t.Run("Should return nav links for starred dashboards", func(t *testing.T) {
|
|
starService := startest.NewStarServiceFake()
|
|
starService.ExpectedUserStars = &star.GetUserStarsResult{
|
|
UserStars: map[string]bool{
|
|
"dashboard1": true,
|
|
"dashboard2": true,
|
|
},
|
|
}
|
|
|
|
dashboardService := dashboards.NewFakeDashboardService(t)
|
|
dashboardService.On("SearchDashboards", context.Background(), mock.Anything).Return(model.HitList{
|
|
{
|
|
UID: "dashboard1",
|
|
Title: "Dashboard 1",
|
|
URL: "/d/dashboard1/",
|
|
},
|
|
{
|
|
UID: "dashboard2",
|
|
Title: "Dashboard 2",
|
|
URL: "/d/dashboard2/",
|
|
},
|
|
}, nil)
|
|
|
|
service := ServiceImpl{
|
|
starService: starService,
|
|
dashboardService: dashboardService,
|
|
}
|
|
|
|
navLinks, err := service.buildStarredItemsNavLinks(reqCtx)
|
|
require.NoError(t, err)
|
|
require.Len(t, navLinks, 2)
|
|
|
|
require.Equal(t, "starred/dashboard1", navLinks[0].Id)
|
|
require.Equal(t, "Dashboard 1", navLinks[0].Text)
|
|
require.Equal(t, "/d/dashboard1/", navLinks[0].Url)
|
|
require.Equal(t, "starred/dashboard2", navLinks[1].Id)
|
|
require.Equal(t, "Dashboard 2", navLinks[1].Text)
|
|
require.Equal(t, "/d/dashboard2/", navLinks[1].Url)
|
|
})
|
|
|
|
t.Run("Should limit to 50 starred dashboards", func(t *testing.T) {
|
|
starService := startest.NewStarServiceFake()
|
|
userStars := make(map[string]bool)
|
|
for i := 0; i < 60; i++ {
|
|
userStars[fmt.Sprintf("dashboard%d", i)] = true
|
|
}
|
|
starService.ExpectedUserStars = &star.GetUserStarsResult{
|
|
UserStars: userStars,
|
|
}
|
|
|
|
dashboardList := make(model.HitList, 60)
|
|
for i := 0; i < 60; i++ {
|
|
dashboardList[i] = &model.Hit{
|
|
UID: fmt.Sprintf("dashboard%d", i),
|
|
Title: fmt.Sprintf("Dashboard %d", i),
|
|
URL: fmt.Sprintf("/d/dashboard%d/", i),
|
|
}
|
|
}
|
|
|
|
dashboardService := dashboards.NewFakeDashboardService(t)
|
|
dashboardService.On("SearchDashboards", context.Background(), mock.Anything).Return(dashboardList, nil)
|
|
|
|
service := ServiceImpl{
|
|
starService: starService,
|
|
dashboardService: dashboardService,
|
|
}
|
|
|
|
navLinks, err := service.buildStarredItemsNavLinks(reqCtx)
|
|
require.NoError(t, err)
|
|
require.Len(t, navLinks, 50)
|
|
})
|
|
|
|
t.Run("Should sort dashboards by title", func(t *testing.T) {
|
|
starService := startest.NewStarServiceFake()
|
|
starService.ExpectedUserStars = &star.GetUserStarsResult{
|
|
UserStars: map[string]bool{
|
|
"dashboard1": true,
|
|
"dashboard2": true,
|
|
"dashboard3": true,
|
|
},
|
|
}
|
|
|
|
dashboardService := dashboards.NewFakeDashboardService(t)
|
|
dashboardService.On("SearchDashboards", context.Background(), mock.Anything).Return(model.HitList{
|
|
{
|
|
UID: "dashboard1",
|
|
Title: "C Dashboard",
|
|
URL: "/d/dashboard1/",
|
|
},
|
|
{
|
|
UID: "dashboard2",
|
|
Title: "A Dashboard",
|
|
URL: "/d/dashboard2/",
|
|
},
|
|
{
|
|
UID: "dashboard3",
|
|
Title: "B Dashboard",
|
|
URL: "/d/dashboard3/",
|
|
},
|
|
}, nil)
|
|
|
|
service := ServiceImpl{
|
|
starService: starService,
|
|
dashboardService: dashboardService,
|
|
}
|
|
|
|
navLinks, err := service.buildStarredItemsNavLinks(reqCtx)
|
|
require.NoError(t, err)
|
|
require.Len(t, navLinks, 3)
|
|
|
|
require.Equal(t, "A Dashboard", navLinks[0].Text)
|
|
require.Equal(t, "B Dashboard", navLinks[1].Text)
|
|
require.Equal(t, "C Dashboard", navLinks[2].Text)
|
|
})
|
|
}
|
|
|
|
func TestHasConnectionsPluginItems(t *testing.T) {
|
|
httpReq, _ := http.NewRequest(http.MethodGet, "", nil)
|
|
reqCtx := &contextmodel.ReqContext{
|
|
SignedInUser: &user.SignedInUser{
|
|
UserID: 1,
|
|
OrgID: 1,
|
|
},
|
|
Context: &web.Context{Req: httpReq},
|
|
}
|
|
|
|
t.Run("Should return false when plugin item exists but user lacks RBAC permission", func(t *testing.T) {
|
|
accessControl := actest.FakeAccessControl{
|
|
ExpectedEvaluate: false, // No RBAC action access
|
|
}
|
|
mockPluginStore := &pluginstore.FakePluginStore{
|
|
PluginList: []pluginstore.Plugin{
|
|
{
|
|
JSONData: plugins.JSONData{
|
|
ID: "grafana-pdc-app",
|
|
Includes: []*plugins.Includes{
|
|
{
|
|
Type: "page",
|
|
Path: "/connections/private-data-source-connections",
|
|
Action: "grafana-pdc-app.private-networks:read",
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
service := ServiceImpl{
|
|
accessControl: &accessControl,
|
|
pluginStore: mockPluginStore,
|
|
}
|
|
|
|
treeRoot := &navtree.NavTreeRoot{
|
|
Children: []*navtree.NavLink{
|
|
{
|
|
Id: "connections",
|
|
Children: []*navtree.NavLink{
|
|
{
|
|
Id: "standalone-plugin-page-/connections/private-data-source-connections",
|
|
Url: "/connections/private-data-source-connections",
|
|
PluginID: "grafana-pdc-app",
|
|
},
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
result := service.hasConnectionsPluginItems(reqCtx, treeRoot)
|
|
require.False(t, result)
|
|
})
|
|
|
|
t.Run("Should return true when user has RBAC permission", func(t *testing.T) {
|
|
accessControl := actest.FakeAccessControl{
|
|
ExpectedEvaluate: true, // Has RBAC access
|
|
}
|
|
mockPluginStore := &pluginstore.FakePluginStore{
|
|
PluginList: []pluginstore.Plugin{
|
|
{
|
|
JSONData: plugins.JSONData{
|
|
ID: "grafana-pdc-app",
|
|
Includes: []*plugins.Includes{
|
|
{
|
|
Type: "page",
|
|
Path: "/connections/private-data-source-connections",
|
|
Action: "grafana-pdc-app.private-networks:read",
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
service := ServiceImpl{
|
|
accessControl: &accessControl,
|
|
pluginStore: mockPluginStore,
|
|
}
|
|
|
|
treeRoot := &navtree.NavTreeRoot{
|
|
Children: []*navtree.NavLink{
|
|
{
|
|
Id: "connections",
|
|
Children: []*navtree.NavLink{
|
|
{
|
|
Id: "standalone-plugin-page-/connections/private-data-source-connections",
|
|
Url: "/connections/private-data-source-connections",
|
|
PluginID: "grafana-pdc-app",
|
|
},
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
result := service.hasConnectionsPluginItems(reqCtx, treeRoot)
|
|
require.True(t, result)
|
|
})
|
|
}
|
|
|
|
func TestBuildDataConnectionsNavLink(t *testing.T) {
|
|
httpReq, _ := http.NewRequest(http.MethodGet, "", nil)
|
|
reqCtx := &contextmodel.ReqContext{
|
|
SignedInUser: &user.SignedInUser{
|
|
UserID: 1,
|
|
OrgID: 1,
|
|
},
|
|
Context: &web.Context{Req: httpReq},
|
|
}
|
|
|
|
t.Run("Should return nil when no access and no plugin items", func(t *testing.T) {
|
|
accessControl := actest.FakeAccessControl{}
|
|
mockPluginStore := &pluginstore.FakePluginStore{}
|
|
|
|
service := ServiceImpl{
|
|
cfg: &setting.Cfg{AppSubURL: ""},
|
|
accessControl: &accessControl,
|
|
pluginStore: mockPluginStore,
|
|
}
|
|
|
|
treeRoot := &navtree.NavTreeRoot{}
|
|
|
|
result := service.buildDataConnectionsNavLink(reqCtx, treeRoot)
|
|
require.Nil(t, result)
|
|
})
|
|
|
|
t.Run("Should return connections nav with datasource items when user has access", func(t *testing.T) {
|
|
accessControl := actest.FakeAccessControl{
|
|
ExpectedEvaluate: true,
|
|
}
|
|
mockPluginStore := &pluginstore.FakePluginStore{}
|
|
|
|
service := ServiceImpl{
|
|
cfg: &setting.Cfg{AppSubURL: ""},
|
|
accessControl: &accessControl,
|
|
pluginStore: mockPluginStore,
|
|
}
|
|
|
|
treeRoot := &navtree.NavTreeRoot{}
|
|
|
|
result := service.buildDataConnectionsNavLink(reqCtx, treeRoot)
|
|
require.NotNil(t, result)
|
|
require.Equal(t, "Connections", result.Text)
|
|
require.Equal(t, "connections", result.Id)
|
|
require.Len(t, result.Children, 2)
|
|
})
|
|
|
|
t.Run("Should return connections nav when plugin items exist", func(t *testing.T) {
|
|
accessControl := actest.FakeAccessControl{
|
|
ExpectedEvaluate: false, // No datasources access
|
|
}
|
|
mockPluginStore := &pluginstore.FakePluginStore{}
|
|
|
|
service := ServiceImpl{
|
|
cfg: &setting.Cfg{AppSubURL: ""},
|
|
accessControl: &accessControl,
|
|
pluginStore: mockPluginStore,
|
|
}
|
|
|
|
// TreeRoot with plugin items (non-plugin item for simplicity)
|
|
treeRoot := &navtree.NavTreeRoot{
|
|
Children: []*navtree.NavLink{
|
|
{
|
|
Id: "some-section",
|
|
Children: []*navtree.NavLink{
|
|
{
|
|
Id: "standalone-plugin-page-/connections/infrastructure",
|
|
Url: "/connections/infrastructure",
|
|
// No PluginID - non-plugin item
|
|
},
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
result := service.buildDataConnectionsNavLink(reqCtx, treeRoot)
|
|
require.NotNil(t, result)
|
|
require.Equal(t, "Connections", result.Text)
|
|
require.Empty(t, result.Children) // No datasource items, but section exists for plugins
|
|
})
|
|
}
|