mirror of
https://github.com/grafana/grafana.git
synced 2025-01-16 11:42:35 -06:00
17ca61d7f8
* Folders: Optionally include fullpath in service responses * Alerting: Export folder fullpath instead of title * Escape separator in folder title * Add support for provisiong alret rules into subfolders * Use FolderService for creating folders during provisioning * Export WithFullpath() folder service function --------- Co-authored-by: Tania B <yalyna.ts@gmail.com> Co-authored-by: Yuri Tseretyan <yuriy.tseretyan@grafana.com>
178 lines
7.2 KiB
Go
178 lines
7.2 KiB
Go
package api
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"net/http"
|
|
|
|
"github.com/grafana/grafana/pkg/api/response"
|
|
contextmodel "github.com/grafana/grafana/pkg/services/contexthandler/model"
|
|
|
|
authz "github.com/grafana/grafana/pkg/services/ngalert/accesscontrol"
|
|
apimodels "github.com/grafana/grafana/pkg/services/ngalert/api/tooling/definitions"
|
|
ngmodels "github.com/grafana/grafana/pkg/services/ngalert/models"
|
|
)
|
|
|
|
// ExportFromPayload converts the rule groups from the argument `ruleGroupConfig` to export format. All rules are expected to be fully specified. The access to data sources mentioned in the rules is not enforced.
|
|
// Can return 403 StatusForbidden if user is not authorized to read folder `namespaceUID`
|
|
func (srv RulerSrv) ExportFromPayload(c *contextmodel.ReqContext, ruleGroupConfig apimodels.PostableRuleGroupConfig, namespaceUID string) response.Response {
|
|
namespace, err := srv.store.GetNamespaceByUID(c.Req.Context(), namespaceUID, c.SignedInUser.GetOrgID(), c.SignedInUser)
|
|
if err != nil {
|
|
return toNamespaceErrorResponse(err)
|
|
}
|
|
|
|
rulesWithOptionals, err := ValidateRuleGroup(&ruleGroupConfig, c.SignedInUser.GetOrgID(), namespace.UID, RuleLimitsFromConfig(srv.cfg, srv.featureManager))
|
|
if err != nil {
|
|
return ErrResp(http.StatusBadRequest, err, "")
|
|
}
|
|
|
|
if len(rulesWithOptionals) == 0 {
|
|
return ErrResp(http.StatusBadRequest, err, "")
|
|
}
|
|
|
|
rules := make([]ngmodels.AlertRule, 0, len(rulesWithOptionals))
|
|
for _, optional := range rulesWithOptionals {
|
|
rules = append(rules, optional.AlertRule)
|
|
}
|
|
|
|
groupsWithFullpath := ngmodels.NewAlertRuleGroupWithFolderFullpath(rules[0].GetGroupKey(), rules, namespace.Fullpath)
|
|
|
|
e, err := AlertingFileExportFromAlertRuleGroupWithFolderFullpath([]ngmodels.AlertRuleGroupWithFolderFullpath{groupsWithFullpath})
|
|
if err != nil {
|
|
return ErrResp(http.StatusInternalServerError, err, "failed to create alerting file export")
|
|
}
|
|
|
|
return exportResponse(c, e)
|
|
}
|
|
|
|
// ExportRules reads alert rules that user has access to from database according to the filters.
|
|
func (srv RulerSrv) ExportRules(c *contextmodel.ReqContext) response.Response {
|
|
// The similar method exists in provisioning (see ProvisioningSrv.RouteGetAlertRulesExport).
|
|
// Modification to parameters and response format should be made in these two methods at the same time.
|
|
|
|
folderUIDs := c.QueryStrings("folderUid")
|
|
group := c.Query("group")
|
|
uid := c.Query("ruleUid")
|
|
|
|
var groups []ngmodels.AlertRuleGroupWithFolderFullpath
|
|
if uid != "" {
|
|
if group != "" || len(folderUIDs) > 0 {
|
|
return ErrResp(http.StatusBadRequest, errors.New("group and folder should not be specified when a single rule is requested"), "")
|
|
}
|
|
rulesGroup, err := srv.getRuleWithFolderFullpathByRuleUid(c, uid)
|
|
if err != nil {
|
|
return errorToResponse(err)
|
|
}
|
|
groups = []ngmodels.AlertRuleGroupWithFolderFullpath{rulesGroup}
|
|
} else if group != "" {
|
|
if len(folderUIDs) != 1 || folderUIDs[0] == "" {
|
|
return ErrResp(http.StatusBadRequest,
|
|
fmt.Errorf("group name must be specified together with a single folder_uid parameter. Got %d", len(folderUIDs)),
|
|
"",
|
|
)
|
|
}
|
|
rulesGroup, err := srv.getRuleGroupWithFolderFullPath(c, ngmodels.AlertRuleGroupKey{
|
|
OrgID: c.SignedInUser.GetOrgID(),
|
|
NamespaceUID: folderUIDs[0],
|
|
RuleGroup: group,
|
|
})
|
|
if err != nil {
|
|
return errorToResponse(err)
|
|
}
|
|
groups = []ngmodels.AlertRuleGroupWithFolderFullpath{rulesGroup}
|
|
} else {
|
|
var err error
|
|
groups, err = srv.getRulesWithFolderFullPathInFolders(c, folderUIDs)
|
|
if err != nil {
|
|
return errorToResponse(err)
|
|
}
|
|
}
|
|
|
|
if len(groups) == 0 {
|
|
return response.Empty(http.StatusNotFound)
|
|
}
|
|
|
|
// sort result so the response is always stable
|
|
ngmodels.SortAlertRuleGroupWithFolderTitle(groups)
|
|
|
|
e, err := AlertingFileExportFromAlertRuleGroupWithFolderFullpath(groups)
|
|
if err != nil {
|
|
return ErrResp(http.StatusInternalServerError, err, "failed to create alerting file export")
|
|
}
|
|
return exportResponse(c, e)
|
|
}
|
|
|
|
// getRuleWithFolderFullpathByRuleUid calls getAuthorizedRuleByUid and combines its result with folder (aka namespace) title.
|
|
func (srv RulerSrv) getRuleWithFolderFullpathByRuleUid(c *contextmodel.ReqContext, ruleUID string) (ngmodels.AlertRuleGroupWithFolderFullpath, error) {
|
|
rule, err := srv.getAuthorizedRuleByUid(c.Req.Context(), c, ruleUID)
|
|
if err != nil {
|
|
return ngmodels.AlertRuleGroupWithFolderFullpath{}, err
|
|
}
|
|
namespace, err := srv.store.GetNamespaceByUID(c.Req.Context(), rule.NamespaceUID, c.SignedInUser.GetOrgID(), c.SignedInUser)
|
|
if err != nil {
|
|
return ngmodels.AlertRuleGroupWithFolderFullpath{}, errors.Join(errFolderAccess, err)
|
|
}
|
|
return ngmodels.NewAlertRuleGroupWithFolderFullpath(rule.GetGroupKey(), []ngmodels.AlertRule{rule}, namespace.Fullpath), nil
|
|
}
|
|
|
|
// getRuleGroupWithFolderFullPath calls getAuthorizedRuleGroup and combines its result with folder (aka namespace) title.
|
|
func (srv RulerSrv) getRuleGroupWithFolderFullPath(c *contextmodel.ReqContext, ruleGroupKey ngmodels.AlertRuleGroupKey) (ngmodels.AlertRuleGroupWithFolderFullpath, error) {
|
|
namespace, err := srv.store.GetNamespaceByUID(c.Req.Context(), ruleGroupKey.NamespaceUID, c.SignedInUser.GetOrgID(), c.SignedInUser)
|
|
if err != nil {
|
|
return ngmodels.AlertRuleGroupWithFolderFullpath{}, errors.Join(errFolderAccess, err)
|
|
}
|
|
rules, err := srv.getAuthorizedRuleGroup(c.Req.Context(), c, ruleGroupKey)
|
|
if err != nil {
|
|
return ngmodels.AlertRuleGroupWithFolderFullpath{}, err
|
|
}
|
|
if len(rules) == 0 {
|
|
return ngmodels.AlertRuleGroupWithFolderFullpath{}, ngmodels.ErrAlertRuleNotFound
|
|
}
|
|
return ngmodels.NewAlertRuleGroupWithFolderFullpathFromRulesGroup(ruleGroupKey, rules, namespace.Fullpath), nil
|
|
}
|
|
|
|
// getRulesWithFolderFullPathInFolders gets list of folders to which user has access, and then calls searchAuthorizedAlertRules.
|
|
// If argument folderUIDs is not empty it intersects it with the list of folders available for user and then retrieves rules that are in those folders.
|
|
func (srv RulerSrv) getRulesWithFolderFullPathInFolders(c *contextmodel.ReqContext, folderUIDs []string) ([]ngmodels.AlertRuleGroupWithFolderFullpath, error) {
|
|
folders, err := srv.store.GetUserVisibleNamespaces(c.Req.Context(), c.SignedInUser.GetOrgID(), c.SignedInUser)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
query := ngmodels.ListAlertRulesQuery{
|
|
OrgID: c.SignedInUser.GetOrgID(),
|
|
NamespaceUIDs: nil,
|
|
}
|
|
if len(folderUIDs) > 0 {
|
|
for _, folderUID := range folderUIDs {
|
|
if _, ok := folders[folderUID]; ok {
|
|
query.NamespaceUIDs = append(query.NamespaceUIDs, folderUID)
|
|
}
|
|
}
|
|
if len(query.NamespaceUIDs) == 0 {
|
|
return nil, authz.NewAuthorizationErrorGeneric("access rules in the specified folders")
|
|
}
|
|
} else {
|
|
for _, folder := range folders {
|
|
query.NamespaceUIDs = append(query.NamespaceUIDs, folder.UID)
|
|
}
|
|
}
|
|
|
|
rulesByGroup, _, err := srv.searchAuthorizedAlertRules(c.Req.Context(), authorizedRuleGroupQuery{
|
|
User: c.SignedInUser,
|
|
NamespaceUIDs: folderUIDs,
|
|
})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
result := make([]ngmodels.AlertRuleGroupWithFolderFullpath, 0, len(rulesByGroup))
|
|
for groupKey, rulesGroup := range rulesByGroup {
|
|
namespace, ok := folders[groupKey.NamespaceUID]
|
|
if !ok {
|
|
continue // user does not have access
|
|
}
|
|
result = append(result, ngmodels.NewAlertRuleGroupWithFolderFullpathFromRulesGroup(groupKey, rulesGroup, namespace.Fullpath))
|
|
}
|
|
return result, nil
|
|
}
|