| 
									
										
										
										
											2021-03-12 03:28:00 +08:00
										 |  |  | package api | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							| 
									
										
										
										
											2021-03-24 19:43:25 +08:00
										 |  |  | 	"bytes" | 
					
						
							|  |  |  | 	"encoding/json" | 
					
						
							| 
									
										
										
										
											2021-03-12 03:28:00 +08:00
										 |  |  | 	"fmt" | 
					
						
							| 
									
										
										
										
											2021-03-24 19:43:25 +08:00
										 |  |  | 	"net/http" | 
					
						
							| 
									
										
										
										
											2021-03-12 03:28:00 +08:00
										 |  |  | 	"regexp" | 
					
						
							| 
									
										
										
										
											2021-03-24 19:43:25 +08:00
										 |  |  | 	"strconv" | 
					
						
							| 
									
										
										
										
											2021-03-17 18:47:03 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/go-openapi/strfmt" | 
					
						
							| 
									
										
										
										
											2021-03-24 19:43:25 +08:00
										 |  |  | 	apimodels "github.com/grafana/alerting-api/pkg/api" | 
					
						
							|  |  |  | 	"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" | 
					
						
							|  |  |  | 	"gopkg.in/macaron.v1" | 
					
						
							|  |  |  | 	"gopkg.in/yaml.v3" | 
					
						
							| 
									
										
										
										
											2021-03-12 03:28:00 +08:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 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)) | 
					
						
							|  |  |  | 	})) | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-03-17 18:47:03 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | func timePtr(t strfmt.DateTime) *strfmt.DateTime { | 
					
						
							|  |  |  | 	return &t | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func stringPtr(s string) *string { | 
					
						
							|  |  |  | 	return &s | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-24 19:43:25 +08:00
										 |  |  | 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 | 
					
						
							| 
									
										
										
										
											2021-03-29 23:18:25 +08:00
										 |  |  | 			case "grafana-alertmanager-datasource": | 
					
						
							|  |  |  | 				return apimodels.AlertmanagerBackend, nil | 
					
						
							| 
									
										
										
										
											2021-03-24 19:43:25 +08:00
										 |  |  | 			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, | 
					
						
							|  |  |  | 	req *http.Request, | 
					
						
							|  |  |  | 	extractor func([]byte) (interface{}, error), | 
					
						
							|  |  |  | ) response.Response { | 
					
						
							|  |  |  | 	newCtx, resp := replacedResponseWriter(ctx) | 
					
						
							|  |  |  | 	newCtx.Req.Request = req | 
					
						
							|  |  |  | 	p.DataProxy.ProxyDatasourceRequestWithID(newCtx, ctx.ParamsInt64("Recipient")) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	status := resp.Status() | 
					
						
							|  |  |  | 	if status >= 400 { | 
					
						
							|  |  |  | 		return response.Error(status, string(resp.Body()), 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 | 
					
						
							|  |  |  | } |