e37a780e14
* init
* autogens AM route
* POST dashboards/db spec
* POST alert-notifications spec
* fix description
* re inits vendor, updates grafana to master
* go mod updates
* alerting routes
* renames to receivers
* prometheus endpoints
* align config endpoint with cortex, include templates
* Change grafana receiver type
* Update receivers.go
* rename struct to stop swagger thrashing
* add rules API
* index html
* standalone swagger ui html page
* Update README.md
* Expose GrafanaManagedAlert properties
* Some fixes
- /api/v1/rules/{Namespace} should return a map
- update ExtendedUpsertAlertDefinitionCommand properties
* am alerts routes
* rename prom swagger section for clarity, remove example endpoints
* Add missing json and yaml tags
* folder perms
* make folders POST again
* fix grafana receiver type
* rename fodler->namespace for perms
* make ruler json again
* PR fixes
* silences
* fix Ok -> Ack
* Add id to POST /api/v1/silences (#9)
Signed-off-by: Ganesh Vernekar <cs15btech11018@iith.ac.in>
* Add POST /api/v1/alerts (#10)
Signed-off-by: Ganesh Vernekar <cs15btech11018@iith.ac.in>
* fix silences
* Add testing endpoints
* removes grpc replace directives
* [wip] starts validation
* pkg cleanup
* go mod tidy
* ignores vendor dir
* Change response type for Cortex/Loki alerts
* receiver unmarshaling tests
* ability to split routes between AM & Grafana
* api marshaling & validation
* begins work on routing lib
* [hack] ignores embedded field in generation
* path specific datasource for alerting
* align endpoint names with cloud
* single route per Alerting config
* removes unused routing pkg
* regens spec
* adds datasource param to ruler/prom route paths
* Modifications for supporting migration
* Apply suggestions from code review
* hack for cleaning circular refs in swagger definition
* generates files
* minor fixes for prom endpoints
* decorate prom apis with required: true where applicable
* Revert "generates files"
This reverts commit ef7e975584.
* removes server autogen
* Update imported structs from ngalert
* Fix listing rules response
* Update github.com/prometheus/common dependency
* Update get silence response
* Update get silences response
* adds ruler validation & backend switching
* Fix GET /alertmanager/{DatasourceId}/config/api/v1/alerts response
* Distinct gettable and postable grafana receivers
* Remove permissions routes
* Latest JSON specs
* Fix testing routes
* inline yaml annotation on apirulenode
* yaml test & yamlv3 + comments
* Fix yaml annotations for embedded type
* Rename DatasourceId path parameter
* Implement Backend.String()
* backend zero value is a real backend
* exports DiscoveryBase
* Fix GO initialisms
* Silences: Use PostableSilence as the base struct for creating silences
* Use type alias instead of struct embedding
* More fixes to alertmanager silencing routes
* post and spec JSONs
* Split rule config to postable/gettable
* Fix empty POST /silences payload
Recreating the generated JSON specs fixes the issue
without further modifications
* better yaml unmarshaling for nested yaml docs in cortex-am configs
* regens spec
* re-adds config.receivers
* omitempty to align with prometheus API behavior
* Prefix routes with /api
* Update Alertmanager models
* Make adjustments to follow the Alertmanager API
* ruler: add for and annotations to grafana alert (#45)
* Modify testing API routes
* Fix grafana rule for field type
* Move PostableUserConfig validation to this library
* Fix PostableUserConfig YAML encoding/decoding
* Use common fields for grafana and lotex rules
* Add namespace id in GettableGrafanaRule
* Apply suggestions from code review
* fixup
* more changes
* Apply suggestions from code review
* aligns structure pre merge
* fix new imports & tests
* updates tooling readme
* goimports
* lint
* more linting!!
* revive lint
Co-authored-by: Sofia Papagiannaki <papagian@gmail.com>
Co-authored-by: Domas <domasx2@gmail.com>
Co-authored-by: Sofia Papagiannaki <papagian@users.noreply.github.com>
Co-authored-by: Ganesh Vernekar <15064823+codesome@users.noreply.github.com>
Co-authored-by: gotjosh <josue@grafana.com>
Co-authored-by: David Parrott <stomp.box.yo@gmail.com>
Co-authored-by: Kyle Brandt <kyle@grafana.com>
224 lines
6.2 KiB
Go
224 lines
6.2 KiB
Go
package api
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/json"
|
|
"fmt"
|
|
"io"
|
|
"net/http"
|
|
"net/url"
|
|
"regexp"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"github.com/grafana/grafana-plugin-sdk-go/data"
|
|
"github.com/grafana/grafana/pkg/api/response"
|
|
"github.com/grafana/grafana/pkg/models"
|
|
"github.com/grafana/grafana/pkg/services/datasourceproxy"
|
|
"github.com/grafana/grafana/pkg/services/datasources"
|
|
apimodels "github.com/grafana/grafana/pkg/services/ngalert/api/tooling/definitions"
|
|
"github.com/grafana/grafana/pkg/services/ngalert/eval"
|
|
ngmodels "github.com/grafana/grafana/pkg/services/ngalert/models"
|
|
"github.com/grafana/grafana/pkg/setting"
|
|
"github.com/grafana/grafana/pkg/tsdb"
|
|
"github.com/grafana/grafana/pkg/util"
|
|
"gopkg.in/macaron.v1"
|
|
"gopkg.in/yaml.v3"
|
|
)
|
|
|
|
var searchRegex = regexp.MustCompile(`\{(\w+)\}`)
|
|
|
|
func toMacaronPath(path string) string {
|
|
return string(searchRegex.ReplaceAllFunc([]byte(path), func(s []byte) []byte {
|
|
m := string(s[1 : len(s)-1])
|
|
return []byte(fmt.Sprintf(":%s", m))
|
|
}))
|
|
}
|
|
|
|
func backendType(ctx *models.ReqContext, cache datasources.CacheService) (apimodels.Backend, error) {
|
|
recipient := ctx.Params("Recipient")
|
|
if recipient == apimodels.GrafanaBackend.String() {
|
|
return apimodels.GrafanaBackend, nil
|
|
}
|
|
if datasourceID, err := strconv.ParseInt(recipient, 10, 64); err == nil {
|
|
if ds, err := cache.GetDatasource(datasourceID, ctx.SignedInUser, ctx.SkipCache); err == nil {
|
|
switch ds.Type {
|
|
case "loki", "prometheus":
|
|
return apimodels.LoTexRulerBackend, nil
|
|
case "alertmanager":
|
|
return apimodels.AlertmanagerBackend, nil
|
|
default:
|
|
return 0, fmt.Errorf("unexpected backend type (%v)", ds.Type)
|
|
}
|
|
}
|
|
}
|
|
return 0, fmt.Errorf("unexpected backend type (%v)", recipient)
|
|
}
|
|
|
|
// macaron unsafely asserts the http.ResponseWriter is an http.CloseNotifier, which will panic.
|
|
// Here we impl it, which will ensure this no longer happens, but neither will we take
|
|
// advantage cancelling upstream requests when the downstream has closed.
|
|
// NB: http.CloseNotifier is a deprecated ifc from before the context pkg.
|
|
type safeMacaronWrapper struct {
|
|
http.ResponseWriter
|
|
}
|
|
|
|
func (w *safeMacaronWrapper) CloseNotify() <-chan bool {
|
|
return make(chan bool)
|
|
}
|
|
|
|
// replacedResponseWriter overwrites the underlying responsewriter used by a *models.ReqContext.
|
|
// It's ugly because it needs to replace a value behind a few nested pointers.
|
|
func replacedResponseWriter(ctx *models.ReqContext) (*models.ReqContext, *response.NormalResponse) {
|
|
resp := response.CreateNormalResponse(make(http.Header), nil, 0)
|
|
cpy := *ctx
|
|
cpyMCtx := *cpy.Context
|
|
cpyMCtx.Resp = macaron.NewResponseWriter(ctx.Req.Method, &safeMacaronWrapper{resp})
|
|
cpy.Context = &cpyMCtx
|
|
return &cpy, resp
|
|
}
|
|
|
|
type AlertingProxy struct {
|
|
DataProxy *datasourceproxy.DatasourceProxyService
|
|
}
|
|
|
|
// withReq proxies a different request
|
|
func (p *AlertingProxy) withReq(
|
|
ctx *models.ReqContext,
|
|
method string,
|
|
u *url.URL,
|
|
body io.Reader,
|
|
extractor func([]byte) (interface{}, error),
|
|
headers map[string]string,
|
|
) response.Response {
|
|
req, err := http.NewRequest(method, u.String(), body)
|
|
if err != nil {
|
|
return response.Error(400, err.Error(), nil)
|
|
}
|
|
for h, v := range headers {
|
|
req.Header.Add(h, v)
|
|
}
|
|
newCtx, resp := replacedResponseWriter(ctx)
|
|
newCtx.Req.Request = req
|
|
p.DataProxy.ProxyDatasourceRequestWithID(newCtx, ctx.ParamsInt64("Recipient"))
|
|
|
|
status := resp.Status()
|
|
if status >= 400 {
|
|
errMessage := string(resp.Body())
|
|
// if Content-Type is application/json
|
|
// and it is successfully decoded and contains a message
|
|
// return this as response error message
|
|
if strings.HasPrefix(resp.Header().Get("Content-Type"), "application/json") {
|
|
var m map[string]interface{}
|
|
if err := json.Unmarshal(resp.Body(), &m); err == nil {
|
|
if message, ok := m["message"]; ok {
|
|
errMessage = message.(string)
|
|
}
|
|
}
|
|
}
|
|
return response.Error(status, errMessage, nil)
|
|
}
|
|
|
|
t, err := extractor(resp.Body())
|
|
if err != nil {
|
|
return response.Error(500, err.Error(), nil)
|
|
}
|
|
|
|
b, err := json.Marshal(t)
|
|
if err != nil {
|
|
return response.Error(500, err.Error(), nil)
|
|
}
|
|
|
|
return response.JSON(status, b)
|
|
}
|
|
|
|
func yamlExtractor(v interface{}) func([]byte) (interface{}, error) {
|
|
return func(b []byte) (interface{}, error) {
|
|
decoder := yaml.NewDecoder(bytes.NewReader(b))
|
|
decoder.KnownFields(true)
|
|
|
|
err := decoder.Decode(v)
|
|
|
|
return v, err
|
|
}
|
|
}
|
|
|
|
func jsonExtractor(v interface{}) func([]byte) (interface{}, error) {
|
|
if v == nil {
|
|
// json unmarshal expects a pointer
|
|
v = &map[string]interface{}{}
|
|
}
|
|
return func(b []byte) (interface{}, error) {
|
|
return v, json.Unmarshal(b, v)
|
|
}
|
|
}
|
|
|
|
func messageExtractor(b []byte) (interface{}, error) {
|
|
return map[string]string{"message": string(b)}, nil
|
|
}
|
|
|
|
func validateCondition(c ngmodels.Condition, user *models.SignedInUser, skipCache bool, datasourceCache datasources.CacheService) error {
|
|
var refID string
|
|
|
|
if len(c.Data) == 0 {
|
|
return nil
|
|
}
|
|
|
|
for _, query := range c.Data {
|
|
if c.Condition == query.RefID {
|
|
refID = c.Condition
|
|
}
|
|
|
|
datasourceUID, err := query.GetDatasource()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
isExpression, err := query.IsExpression()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if isExpression {
|
|
continue
|
|
}
|
|
|
|
_, err = datasourceCache.GetDatasourceByUID(datasourceUID, user, skipCache)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to get datasource: %s: %w", datasourceUID, err)
|
|
}
|
|
}
|
|
|
|
if refID == "" {
|
|
return fmt.Errorf("condition %s not found in any query or expression", c.Condition)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func conditionEval(c *models.ReqContext, cmd ngmodels.EvalAlertConditionCommand, datasourceCache datasources.CacheService, dataService *tsdb.Service, cfg *setting.Cfg) response.Response {
|
|
evalCond := ngmodels.Condition{
|
|
Condition: cmd.Condition,
|
|
OrgID: c.SignedInUser.OrgId,
|
|
Data: cmd.Data,
|
|
}
|
|
if err := validateCondition(evalCond, c.SignedInUser, c.SkipCache, datasourceCache); err != nil {
|
|
return response.Error(400, "invalid condition", err)
|
|
}
|
|
|
|
now := cmd.Now
|
|
if now.IsZero() {
|
|
now = timeNow()
|
|
}
|
|
|
|
evaluator := eval.Evaluator{Cfg: cfg}
|
|
evalResults, err := evaluator.ConditionEval(&evalCond, timeNow(), dataService)
|
|
if err != nil {
|
|
return response.Error(400, "Failed to evaluate conditions", err)
|
|
}
|
|
|
|
frame := evalResults.AsDataFrame()
|
|
|
|
return response.JSONStreaming(200, util.DynMap{
|
|
"instances": []*data.Frame{&frame},
|
|
})
|
|
}
|