mirror of https://github.com/grafana/grafana
prometheushacktoberfestmetricsmonitoringalertinggrafanagoinfluxdbmysqlpostgresanalyticsdata-visualizationdashboardbusiness-intelligenceelasticsearch
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
599 lines
20 KiB
599 lines
20 KiB
![]()
3 years ago
|
package navtreeimpl
|
||
|
|
||
|
import (
|
||
|
"fmt"
|
||
|
"sort"
|
||
|
|
||
|
"github.com/grafana/grafana/pkg/api/dtos"
|
||
|
"github.com/grafana/grafana/pkg/infra/kvstore"
|
||
|
"github.com/grafana/grafana/pkg/infra/log"
|
||
|
"github.com/grafana/grafana/pkg/models"
|
||
|
"github.com/grafana/grafana/pkg/plugins"
|
||
|
ac "github.com/grafana/grafana/pkg/services/accesscontrol"
|
||
|
"github.com/grafana/grafana/pkg/services/apikey"
|
||
|
"github.com/grafana/grafana/pkg/services/dashboards"
|
||
|
"github.com/grafana/grafana/pkg/services/featuremgmt"
|
||
|
"github.com/grafana/grafana/pkg/services/navtree"
|
||
|
"github.com/grafana/grafana/pkg/services/org"
|
||
|
"github.com/grafana/grafana/pkg/services/pluginsettings"
|
||
|
pref "github.com/grafana/grafana/pkg/services/preference"
|
||
|
"github.com/grafana/grafana/pkg/services/star"
|
||
|
"github.com/grafana/grafana/pkg/setting"
|
||
|
)
|
||
|
|
||
|
type ServiceImpl struct {
|
||
|
cfg *setting.Cfg
|
||
|
log log.Logger
|
||
|
accessControl ac.AccessControl
|
||
|
pluginStore plugins.Store
|
||
|
pluginSettings pluginsettings.Service
|
||
|
starService star.Service
|
||
|
features *featuremgmt.FeatureManager
|
||
|
dashboardService dashboards.DashboardService
|
||
|
accesscontrolService ac.Service
|
||
|
kvStore kvstore.KVStore
|
||
|
apiKeyService apikey.Service
|
||
|
}
|
||
|
|
||
|
func ProvideService(cfg *setting.Cfg, accessControl ac.AccessControl, pluginStore plugins.Store, pluginSettings pluginsettings.Service, starService star.Service, features *featuremgmt.FeatureManager, dashboardService dashboards.DashboardService, accesscontrolService ac.Service, kvStore kvstore.KVStore, apiKeyService apikey.Service) navtree.Service {
|
||
|
return &ServiceImpl{
|
||
|
cfg: cfg,
|
||
|
log: log.New("navtree service"),
|
||
|
accessControl: accessControl,
|
||
|
pluginStore: pluginStore,
|
||
|
pluginSettings: pluginSettings,
|
||
|
starService: starService,
|
||
|
features: features,
|
||
|
dashboardService: dashboardService,
|
||
|
accesscontrolService: accesscontrolService,
|
||
|
kvStore: kvStore,
|
||
|
apiKeyService: apiKeyService,
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//nolint:gocyclo
|
||
|
func (s *ServiceImpl) GetNavTree(c *models.ReqContext, hasEditPerm bool, prefs *pref.Preference) ([]*navtree.NavLink, error) {
|
||
|
hasAccess := ac.HasAccess(s.accessControl, c)
|
||
|
var navTree []*navtree.NavLink
|
||
|
|
||
|
if hasAccess(ac.ReqSignedIn, ac.EvalPermission(dashboards.ActionDashboardsRead)) {
|
||
|
starredItemsLinks, err := s.buildStarredItemsNavLinks(c, prefs)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
navTree = append(navTree, &navtree.NavLink{
|
||
|
Text: "Starred",
|
||
|
Id: "starred",
|
||
|
Icon: "star",
|
||
|
SortWeight: navtree.WeightSavedItems,
|
||
|
Section: navtree.NavSectionCore,
|
||
|
Children: starredItemsLinks,
|
||
|
EmptyMessageId: "starred-empty",
|
||
|
})
|
||
|
|
||
|
dashboardChildLinks := s.buildDashboardNavLinks(c, hasEditPerm)
|
||
|
|
||
|
dashboardsUrl := "/dashboards"
|
||
|
|
||
|
dashboardLink := &navtree.NavLink{
|
||
|
Text: "Dashboards",
|
||
|
Id: "dashboards",
|
||
|
SubTitle: "Manage dashboards and folders",
|
||
|
Icon: "apps",
|
||
|
Url: s.cfg.AppSubURL + dashboardsUrl,
|
||
|
SortWeight: navtree.WeightDashboard,
|
||
|
Section: navtree.NavSectionCore,
|
||
|
Children: dashboardChildLinks,
|
||
|
}
|
||
|
|
||
|
if s.features.IsEnabled(featuremgmt.FlagTopnav) {
|
||
|
dashboardLink.Id = "dashboards/browse"
|
||
|
}
|
||
|
|
||
|
navTree = append(navTree, dashboardLink)
|
||
|
}
|
||
|
|
||
|
canExplore := func(context *models.ReqContext) bool {
|
||
|
return c.OrgRole == org.RoleAdmin || c.OrgRole == org.RoleEditor || setting.ViewersCanEdit
|
||
|
}
|
||
|
|
||
|
if setting.ExploreEnabled && hasAccess(canExplore, ac.EvalPermission(ac.ActionDatasourcesExplore)) {
|
||
|
navTree = append(navTree, &navtree.NavLink{
|
||
|
Text: "Explore",
|
||
|
Id: "explore",
|
||
|
SubTitle: "Explore your data",
|
||
|
Icon: "compass",
|
||
|
SortWeight: navtree.WeightExplore,
|
||
|
Section: navtree.NavSectionCore,
|
||
|
Url: s.cfg.AppSubURL + "/explore",
|
||
|
})
|
||
|
}
|
||
|
|
||
|
navTree = s.addProfile(navTree, c)
|
||
|
|
||
|
_, uaIsDisabledForOrg := s.cfg.UnifiedAlerting.DisabledOrgs[c.OrgID]
|
||
|
uaVisibleForOrg := s.cfg.UnifiedAlerting.IsEnabled() && !uaIsDisabledForOrg
|
||
|
|
||
|
if setting.AlertingEnabled != nil && *setting.AlertingEnabled {
|
||
|
navTree = append(navTree, s.buildLegacyAlertNavLinks(c)...)
|
||
|
} else if uaVisibleForOrg {
|
||
|
navTree = append(navTree, s.buildAlertNavLinks(c, hasEditPerm)...)
|
||
|
}
|
||
|
|
||
|
if s.features.IsEnabled(featuremgmt.FlagDataConnectionsConsole) {
|
||
|
navTree = append(navTree, s.buildDataConnectionsNavLink(c))
|
||
|
}
|
||
|
|
||
|
appLinks, err := s.getAppLinks(c)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
// When topnav is enabled we can test new information architecture where plugins live in Apps category
|
||
|
if s.features.IsEnabled(featuremgmt.FlagTopnav) {
|
||
|
navTree = append(navTree, &navtree.NavLink{
|
||
|
Text: "Apps",
|
||
|
Icon: "apps",
|
||
|
Description: "App plugins",
|
||
|
Id: "apps",
|
||
|
Children: appLinks,
|
||
|
Section: navtree.NavSectionCore,
|
||
|
Url: s.cfg.AppSubURL + "/apps",
|
||
|
})
|
||
|
} else {
|
||
|
navTree = append(navTree, appLinks...)
|
||
|
}
|
||
|
|
||
|
configNodes, err := s.setupConfigNodes(c)
|
||
|
if err != nil {
|
||
|
return navTree, err
|
||
|
}
|
||
|
|
||
|
if s.features.IsEnabled(featuremgmt.FlagLivePipeline) {
|
||
|
liveNavLinks := []*navtree.NavLink{}
|
||
|
|
||
|
liveNavLinks = append(liveNavLinks, &navtree.NavLink{
|
||
|
Text: "Status", Id: "live-status", Url: s.cfg.AppSubURL + "/live", Icon: "exchange-alt",
|
||
|
})
|
||
|
liveNavLinks = append(liveNavLinks, &navtree.NavLink{
|
||
|
Text: "Pipeline", Id: "live-pipeline", Url: s.cfg.AppSubURL + "/live/pipeline", Icon: "arrow-to-right",
|
||
|
})
|
||
|
liveNavLinks = append(liveNavLinks, &navtree.NavLink{
|
||
|
Text: "Cloud", Id: "live-cloud", Url: s.cfg.AppSubURL + "/live/cloud", Icon: "cloud-upload",
|
||
|
})
|
||
|
navTree = append(navTree, &navtree.NavLink{
|
||
|
Id: "live",
|
||
|
Text: "Live",
|
||
|
SubTitle: "Event streaming",
|
||
|
Icon: "exchange-alt",
|
||
|
Url: s.cfg.AppSubURL + "/live",
|
||
|
Children: liveNavLinks,
|
||
|
Section: navtree.NavSectionConfig,
|
||
|
HideFromTabs: true,
|
||
|
})
|
||
|
}
|
||
|
|
||
|
var configNode *navtree.NavLink
|
||
|
var serverAdminNode *navtree.NavLink
|
||
|
|
||
|
if len(configNodes) > 0 {
|
||
|
configNode = &navtree.NavLink{
|
||
|
Id: navtree.NavIDCfg,
|
||
|
Text: "Configuration",
|
||
|
SubTitle: "Organization: " + c.OrgName,
|
||
|
Icon: "cog",
|
||
|
Url: configNodes[0].Url,
|
||
|
Section: navtree.NavSectionConfig,
|
||
|
SortWeight: navtree.WeightConfig,
|
||
|
Children: configNodes,
|
||
|
}
|
||
|
if s.features.IsEnabled(featuremgmt.FlagTopnav) {
|
||
|
configNode.Url = "/admin"
|
||
|
} else {
|
||
|
configNode.Url = configNodes[0].Url
|
||
|
}
|
||
|
navTree = append(navTree, configNode)
|
||
|
}
|
||
|
|
||
|
adminNavLinks := s.buildAdminNavLinks(c)
|
||
|
|
||
|
if len(adminNavLinks) > 0 {
|
||
|
serverAdminNode = navtree.GetServerAdminNode(adminNavLinks)
|
||
|
navTree = append(navTree, serverAdminNode)
|
||
|
}
|
||
|
|
||
|
if s.features.IsEnabled(featuremgmt.FlagTopnav) {
|
||
|
// Move server admin into Configuration and rename to administration
|
||
|
if configNode != nil && serverAdminNode != nil {
|
||
|
configNode.Text = "Administration"
|
||
|
serverAdminNode.Url = "/admin/server"
|
||
|
serverAdminNode.HideFromTabs = false
|
||
|
configNode.Children = append(configNode.Children, serverAdminNode)
|
||
|
adminNodeIndex := len(navTree) - 1
|
||
|
navTree = navTree[:adminNodeIndex]
|
||
|
}
|
||
|
}
|
||
|
|
||
|
navTree = s.addHelpLinks(navTree, c)
|
||
|
|
||
|
return navTree, nil
|
||
|
}
|
||
|
|
||
|
func (s *ServiceImpl) addHelpLinks(navTree []*navtree.NavLink, c *models.ReqContext) []*navtree.NavLink {
|
||
|
if setting.HelpEnabled {
|
||
|
helpVersion := fmt.Sprintf(`%s v%s (%s)`, setting.ApplicationName, setting.BuildVersion, setting.BuildCommit)
|
||
|
if s.cfg.AnonymousHideVersion && !c.IsSignedIn {
|
||
|
helpVersion = setting.ApplicationName
|
||
|
}
|
||
|
|
||
|
navTree = append(navTree, &navtree.NavLink{
|
||
|
Text: "Help",
|
||
|
SubTitle: helpVersion,
|
||
|
Id: "help",
|
||
|
Url: "#",
|
||
|
Icon: "question-circle",
|
||
|
SortWeight: navtree.WeightHelp,
|
||
|
Section: navtree.NavSectionConfig,
|
||
|
Children: []*navtree.NavLink{},
|
||
|
})
|
||
|
}
|
||
|
return navTree
|
||
|
}
|
||
|
|
||
|
func (s *ServiceImpl) addProfile(navTree []*navtree.NavLink, c *models.ReqContext) []*navtree.NavLink {
|
||
|
if setting.ProfileEnabled && c.IsSignedIn {
|
||
|
navTree = append(navTree, s.getProfileNode(c))
|
||
|
}
|
||
|
return navTree
|
||
|
}
|
||
|
|
||
|
func (s *ServiceImpl) getProfileNode(c *models.ReqContext) *navtree.NavLink {
|
||
|
// Only set login if it's different from the name
|
||
|
var login string
|
||
|
if c.SignedInUser.Login != c.SignedInUser.NameOrFallback() {
|
||
|
login = c.SignedInUser.Login
|
||
|
}
|
||
|
gravatarURL := dtos.GetGravatarUrl(c.Email)
|
||
|
|
||
|
children := []*navtree.NavLink{
|
||
|
{
|
||
|
Text: "Preferences", Id: "profile/settings", Url: s.cfg.AppSubURL + "/profile", Icon: "sliders-v-alt",
|
||
|
},
|
||
|
}
|
||
|
|
||
|
children = append(children, &navtree.NavLink{
|
||
|
Text: "Notification history", Id: "profile/notifications", Url: s.cfg.AppSubURL + "/profile/notifications", Icon: "bell",
|
||
|
})
|
||
|
|
||
|
if setting.AddChangePasswordLink() {
|
||
|
children = append(children, &navtree.NavLink{
|
||
|
Text: "Change password", Id: "profile/password", Url: s.cfg.AppSubURL + "/profile/password",
|
||
|
Icon: "lock",
|
||
|
})
|
||
|
}
|
||
|
|
||
|
if !setting.DisableSignoutMenu {
|
||
|
// add sign out first
|
||
|
children = append(children, &navtree.NavLink{
|
||
|
Text: "Sign out",
|
||
|
Id: "sign-out",
|
||
|
Url: s.cfg.AppSubURL + "/logout",
|
||
|
Icon: "arrow-from-right",
|
||
|
Target: "_self",
|
||
|
HideFromTabs: true,
|
||
|
})
|
||
|
}
|
||
|
|
||
|
return &navtree.NavLink{
|
||
|
Text: c.SignedInUser.NameOrFallback(),
|
||
|
SubTitle: login,
|
||
|
Id: "profile",
|
||
|
Img: gravatarURL,
|
||
|
Url: s.cfg.AppSubURL + "/profile",
|
||
|
Section: navtree.NavSectionConfig,
|
||
|
SortWeight: navtree.WeightProfile,
|
||
|
Children: children,
|
||
|
RoundIcon: true,
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func (s *ServiceImpl) buildStarredItemsNavLinks(c *models.ReqContext, prefs *pref.Preference) ([]*navtree.NavLink, error) {
|
||
|
starredItemsChildNavs := []*navtree.NavLink{}
|
||
|
|
||
|
query := star.GetUserStarsQuery{
|
||
|
UserID: c.SignedInUser.UserID,
|
||
|
}
|
||
|
|
||
|
starredDashboardResult, err := s.starService.GetByUser(c.Req.Context(), &query)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
starredDashboards := []*models.Dashboard{}
|
||
|
starredDashboardsCounter := 0
|
||
|
for dashboardId := range starredDashboardResult.UserStars {
|
||
|
// Set a loose limit to the first 50 starred dashboards found
|
||
|
if starredDashboardsCounter > 50 {
|
||
|
break
|
||
|
}
|
||
|
starredDashboardsCounter++
|
||
|
query := &models.GetDashboardQuery{
|
||
|
Id: dashboardId,
|
||
|
OrgId: c.OrgID,
|
||
|
}
|
||
|
err := s.dashboardService.GetDashboard(c.Req.Context(), query)
|
||
|
if err == nil {
|
||
|
starredDashboards = append(starredDashboards, query.Result)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if len(starredDashboards) > 0 {
|
||
|
sort.Slice(starredDashboards, func(i, j int) bool {
|
||
|
return starredDashboards[i].Title < starredDashboards[j].Title
|
||
|
})
|
||
|
for _, starredItem := range starredDashboards {
|
||
|
starredItemsChildNavs = append(starredItemsChildNavs, &navtree.NavLink{
|
||
|
Id: starredItem.Uid,
|
||
|
Text: starredItem.Title,
|
||
|
Url: starredItem.GetUrl(),
|
||
|
})
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return starredItemsChildNavs, nil
|
||
|
}
|
||
|
|
||
|
func (s *ServiceImpl) buildDashboardNavLinks(c *models.ReqContext, hasEditPerm bool) []*navtree.NavLink {
|
||
|
hasAccess := ac.HasAccess(s.accessControl, c)
|
||
|
hasEditPermInAnyFolder := func(c *models.ReqContext) bool {
|
||
|
return hasEditPerm
|
||
|
}
|
||
|
|
||
|
dashboardChildNavs := []*navtree.NavLink{}
|
||
|
if !s.features.IsEnabled(featuremgmt.FlagTopnav) {
|
||
|
dashboardChildNavs = append(dashboardChildNavs, &navtree.NavLink{
|
||
|
Text: "Browse", Id: "dashboards/browse", Url: s.cfg.AppSubURL + "/dashboards", Icon: "sitemap",
|
||
|
})
|
||
|
}
|
||
|
dashboardChildNavs = append(dashboardChildNavs, &navtree.NavLink{
|
||
|
Text: "Playlists", Id: "dashboards/playlists", Url: s.cfg.AppSubURL + "/playlists", Icon: "presentation-play",
|
||
|
})
|
||
|
|
||
|
if c.IsSignedIn {
|
||
|
dashboardChildNavs = append(dashboardChildNavs, &navtree.NavLink{
|
||
|
Text: "Snapshots",
|
||
|
Id: "dashboards/snapshots",
|
||
|
Url: s.cfg.AppSubURL + "/dashboard/snapshots",
|
||
|
Icon: "camera",
|
||
|
})
|
||
|
|
||
|
dashboardChildNavs = append(dashboardChildNavs, &navtree.NavLink{
|
||
|
Text: "Library panels",
|
||
|
Id: "dashboards/library-panels",
|
||
|
Url: s.cfg.AppSubURL + "/library-panels",
|
||
|
Icon: "library-panel",
|
||
|
})
|
||
|
}
|
||
|
|
||
|
if s.features.IsEnabled(featuremgmt.FlagScenes) {
|
||
|
dashboardChildNavs = append(dashboardChildNavs, &navtree.NavLink{
|
||
|
Text: "Scenes",
|
||
|
Id: "scenes",
|
||
|
Url: s.cfg.AppSubURL + "/scenes",
|
||
|
Icon: "apps",
|
||
|
})
|
||
|
}
|
||
|
|
||
|
if hasEditPerm {
|
||
|
dashboardChildNavs = append(dashboardChildNavs, &navtree.NavLink{
|
||
|
Text: "Divider", Divider: true, Id: "divider", HideFromTabs: true,
|
||
|
})
|
||
|
|
||
|
if hasAccess(hasEditPermInAnyFolder, ac.EvalPermission(dashboards.ActionDashboardsCreate)) {
|
||
|
dashboardChildNavs = append(dashboardChildNavs, &navtree.NavLink{
|
||
|
Text: "New dashboard", Icon: "plus", Url: s.cfg.AppSubURL + "/dashboard/new", HideFromTabs: true, Id: "dashboards/new", ShowIconInNavbar: true,
|
||
|
})
|
||
|
}
|
||
|
|
||
|
if hasAccess(ac.ReqOrgAdminOrEditor, ac.EvalPermission(dashboards.ActionFoldersCreate)) {
|
||
|
dashboardChildNavs = append(dashboardChildNavs, &navtree.NavLink{
|
||
|
Text: "New folder", SubTitle: "Create a new folder to organize your dashboards", Id: "dashboards/folder/new",
|
||
|
Icon: "plus", Url: s.cfg.AppSubURL + "/dashboards/folder/new", HideFromTabs: true, ShowIconInNavbar: true,
|
||
|
})
|
||
|
}
|
||
|
|
||
|
if hasAccess(hasEditPermInAnyFolder, ac.EvalPermission(dashboards.ActionDashboardsCreate)) {
|
||
|
dashboardChildNavs = append(dashboardChildNavs, &navtree.NavLink{
|
||
|
Text: "Import", SubTitle: "Import dashboard from file or Grafana.com", Id: "dashboards/import", Icon: "plus",
|
||
|
Url: s.cfg.AppSubURL + "/dashboard/import", HideFromTabs: true, ShowIconInNavbar: true,
|
||
|
})
|
||
|
}
|
||
|
}
|
||
|
return dashboardChildNavs
|
||
|
}
|
||
|
|
||
|
func (s *ServiceImpl) buildLegacyAlertNavLinks(c *models.ReqContext) []*navtree.NavLink {
|
||
|
var alertChildNavs []*navtree.NavLink
|
||
|
alertChildNavs = append(alertChildNavs, &navtree.NavLink{
|
||
|
Text: "Alert rules", Id: "alert-list", Url: s.cfg.AppSubURL + "/alerting/list", Icon: "list-ul",
|
||
|
})
|
||
|
|
||
|
if c.HasRole(org.RoleEditor) {
|
||
|
alertChildNavs = append(alertChildNavs, &navtree.NavLink{
|
||
|
Text: "Notification channels", Id: "channels", Url: s.cfg.AppSubURL + "/alerting/notifications",
|
||
|
Icon: "comment-alt-share",
|
||
|
})
|
||
|
}
|
||
|
|
||
|
var alertNav = navtree.NavLink{
|
||
|
Text: "Alerting",
|
||
|
SubTitle: "Alert rules and notifications",
|
||
|
Id: "alerting-legacy",
|
||
|
Icon: "bell",
|
||
|
Children: alertChildNavs,
|
||
|
Section: navtree.NavSectionCore,
|
||
|
SortWeight: navtree.WeightAlerting,
|
||
|
}
|
||
|
|
||
|
if s.features.IsEnabled(featuremgmt.FlagTopnav) {
|
||
|
alertNav.Url = s.cfg.AppSubURL + "/alerting"
|
||
|
} else {
|
||
|
alertNav.Url = s.cfg.AppSubURL + "/alerting/list"
|
||
|
}
|
||
|
|
||
|
return []*navtree.NavLink{&alertNav}
|
||
|
}
|
||
|
|
||
|
func (s *ServiceImpl) buildAlertNavLinks(c *models.ReqContext, hasEditPerm bool) []*navtree.NavLink {
|
||
|
hasAccess := ac.HasAccess(s.accessControl, c)
|
||
|
var alertChildNavs []*navtree.NavLink
|
||
|
|
||
|
if hasAccess(ac.ReqViewer, ac.EvalAny(ac.EvalPermission(ac.ActionAlertingRuleRead), ac.EvalPermission(ac.ActionAlertingRuleExternalRead))) {
|
||
|
alertChildNavs = append(alertChildNavs, &navtree.NavLink{
|
||
|
Text: "Alert rules", Id: "alert-list", Url: s.cfg.AppSubURL + "/alerting/list", Icon: "list-ul",
|
||
|
})
|
||
|
}
|
||
|
|
||
|
if hasAccess(ac.ReqOrgAdminOrEditor, ac.EvalAny(ac.EvalPermission(ac.ActionAlertingNotificationsRead), ac.EvalPermission(ac.ActionAlertingNotificationsExternalRead))) {
|
||
|
alertChildNavs = append(alertChildNavs, &navtree.NavLink{
|
||
|
Text: "Contact points", Id: "receivers", Url: s.cfg.AppSubURL + "/alerting/notifications",
|
||
|
Icon: "comment-alt-share", SubTitle: "Manage the settings of your contact points",
|
||
|
})
|
||
|
alertChildNavs = append(alertChildNavs, &navtree.NavLink{Text: "Notification policies", Id: "am-routes", Url: s.cfg.AppSubURL + "/alerting/routes", Icon: "sitemap"})
|
||
|
}
|
||
|
|
||
|
if hasAccess(ac.ReqViewer, ac.EvalAny(ac.EvalPermission(ac.ActionAlertingInstanceRead), ac.EvalPermission(ac.ActionAlertingInstancesExternalRead))) {
|
||
|
alertChildNavs = append(alertChildNavs, &navtree.NavLink{Text: "Silences", Id: "silences", Url: s.cfg.AppSubURL + "/alerting/silences", Icon: "bell-slash"})
|
||
|
alertChildNavs = append(alertChildNavs, &navtree.NavLink{Text: "Alert groups", Id: "groups", Url: s.cfg.AppSubURL + "/alerting/groups", Icon: "layer-group"})
|
||
|
}
|
||
|
|
||
|
if c.OrgRole == org.RoleAdmin {
|
||
|
alertChildNavs = append(alertChildNavs, &navtree.NavLink{
|
||
|
Text: "Admin", Id: "alerting-admin", Url: s.cfg.AppSubURL + "/alerting/admin",
|
||
|
Icon: "cog",
|
||
|
})
|
||
|
}
|
||
|
|
||
|
fallbackHasEditPerm := func(*models.ReqContext) bool { return hasEditPerm }
|
||
|
|
||
|
if hasAccess(fallbackHasEditPerm, ac.EvalAny(ac.EvalPermission(ac.ActionAlertingRuleCreate), ac.EvalPermission(ac.ActionAlertingRuleExternalWrite))) {
|
||
|
alertChildNavs = append(alertChildNavs, &navtree.NavLink{
|
||
|
Text: "Divider", Divider: true, Id: "divider", HideFromTabs: true,
|
||
|
})
|
||
|
|
||
|
alertChildNavs = append(alertChildNavs, &navtree.NavLink{
|
||
|
Text: "New alert rule", SubTitle: "Create an alert rule", Id: "alert",
|
||
|
Icon: "plus", Url: s.cfg.AppSubURL + "/alerting/new", HideFromTabs: true, ShowIconInNavbar: true,
|
||
|
})
|
||
|
}
|
||
|
|
||
|
if len(alertChildNavs) > 0 {
|
||
|
var alertNav = navtree.NavLink{
|
||
|
Text: "Alerting",
|
||
|
SubTitle: "Alert rules and notifications",
|
||
|
Id: "alerting",
|
||
|
Icon: "bell",
|
||
|
Children: alertChildNavs,
|
||
|
Section: navtree.NavSectionCore,
|
||
|
SortWeight: navtree.WeightAlerting,
|
||
|
}
|
||
|
|
||
|
if s.features.IsEnabled(featuremgmt.FlagTopnav) {
|
||
|
alertNav.Url = s.cfg.AppSubURL + "/alerting"
|
||
|
} else {
|
||
|
alertNav.Url = s.cfg.AppSubURL + "/alerting/list"
|
||
|
}
|
||
|
|
||
|
return []*navtree.NavLink{&alertNav}
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (s *ServiceImpl) buildDataConnectionsNavLink(c *models.ReqContext) *navtree.NavLink {
|
||
|
var children []*navtree.NavLink
|
||
|
var navLink *navtree.NavLink
|
||
|
|
||
|
baseId := "data-connections"
|
||
|
baseUrl := s.cfg.AppSubURL + "/" + baseId
|
||
|
|
||
|
children = append(children, &navtree.NavLink{
|
||
|
Id: baseId + "-datasources",
|
||
|
Text: "Data sources",
|
||
|
Icon: "database",
|
||
|
Description: "Add and configure data sources",
|
||
|
Url: baseUrl + "/datasources",
|
||
|
})
|
||
|
|
||
|
children = append(children, &navtree.NavLink{
|
||
|
Id: baseId + "-plugins",
|
||
|
Text: "Plugins",
|
||
|
Icon: "plug",
|
||
|
Description: "Manage plugins",
|
||
|
Url: baseUrl + "/plugins",
|
||
|
})
|
||
|
|
||
|
children = append(children, &navtree.NavLink{
|
||
|
Id: baseId + "-cloud-integrations",
|
||
|
Text: "Cloud integrations",
|
||
|
Icon: "bolt",
|
||
|
Description: "Manage your cloud integrations",
|
||
|
Url: baseUrl + "/cloud-integrations",
|
||
|
})
|
||
|
|
||
|
navLink = &navtree.NavLink{
|
||
|
Text: "Data Connections",
|
||
|
Icon: "link",
|
||
|
Id: baseId,
|
||
|
Url: baseUrl,
|
||
|
Children: children,
|
||
|
Section: navtree.NavSectionCore,
|
||
|
SortWeight: navtree.WeightDataConnections,
|
||
|
}
|
||
|
|
||
|
return navLink
|
||
|
}
|
||
|
|
||
|
func (s *ServiceImpl) buildAdminNavLinks(c *models.ReqContext) []*navtree.NavLink {
|
||
|
hasAccess := ac.HasAccess(s.accessControl, c)
|
||
|
hasGlobalAccess := ac.HasGlobalAccess(s.accessControl, s.accesscontrolService, c)
|
||
|
orgsAccessEvaluator := ac.EvalPermission(ac.ActionOrgsRead)
|
||
|
adminNavLinks := []*navtree.NavLink{}
|
||
|
|
||
|
if hasAccess(ac.ReqGrafanaAdmin, ac.EvalPermission(ac.ActionUsersRead, ac.ScopeGlobalUsersAll)) {
|
||
|
adminNavLinks = append(adminNavLinks, &navtree.NavLink{
|
||
|
Text: "Users", Id: "global-users", Url: s.cfg.AppSubURL + "/admin/users", Icon: "user",
|
||
|
})
|
||
|
}
|
||
|
|
||
|
if hasGlobalAccess(ac.ReqGrafanaAdmin, orgsAccessEvaluator) {
|
||
|
adminNavLinks = append(adminNavLinks, &navtree.NavLink{
|
||
|
Text: "Orgs", Id: "global-orgs", Url: s.cfg.AppSubURL + "/admin/orgs", Icon: "building",
|
||
|
})
|
||
|
}
|
||
|
|
||
|
if hasAccess(ac.ReqGrafanaAdmin, ac.EvalPermission(ac.ActionSettingsRead)) {
|
||
|
adminNavLinks = append(adminNavLinks, &navtree.NavLink{
|
||
|
Text: "Settings", Id: "server-settings", Url: s.cfg.AppSubURL + "/admin/settings", Icon: "sliders-v-alt",
|
||
|
})
|
||
|
}
|
||
|
|
||
|
if hasAccess(ac.ReqGrafanaAdmin, ac.EvalPermission(ac.ActionSettingsRead)) && s.features.IsEnabled(featuremgmt.FlagStorage) {
|
||
|
adminNavLinks = append(adminNavLinks, &navtree.NavLink{
|
||
|
Text: "Storage",
|
||
|
Id: "storage",
|
||
|
Description: "Manage file storage",
|
||
|
Icon: "cube",
|
||
|
Url: s.cfg.AppSubURL + "/admin/storage",
|
||
|
})
|
||
|
}
|
||
|
|
||
|
if s.cfg.LDAPEnabled && hasAccess(ac.ReqGrafanaAdmin, ac.EvalPermission(ac.ActionLDAPStatusRead)) {
|
||
|
adminNavLinks = append(adminNavLinks, &navtree.NavLink{
|
||
|
Text: "LDAP", Id: "ldap", Url: s.cfg.AppSubURL + "/admin/ldap", Icon: "book",
|
||
|
})
|
||
|
}
|
||
|
|
||
|
return adminNavLinks
|
||
|
}
|