mirror of
https://github.com/grafana/grafana.git
synced 2024-12-02 13:39:19 -06:00
178 lines
4.4 KiB
Go
178 lines
4.4 KiB
Go
package mqe
|
|
|
|
import (
|
|
"encoding/json"
|
|
"io/ioutil"
|
|
"net/http"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"fmt"
|
|
|
|
"regexp"
|
|
|
|
"github.com/grafana/grafana/pkg/components/null"
|
|
"github.com/grafana/grafana/pkg/log"
|
|
"github.com/grafana/grafana/pkg/tsdb"
|
|
)
|
|
|
|
func NewResponseParser() *ResponseParser {
|
|
return &ResponseParser{
|
|
log: log.New("tsdb.mqe"),
|
|
}
|
|
}
|
|
|
|
var (
|
|
indexAliasPattern *regexp.Regexp
|
|
wildcardAliasPattern *regexp.Regexp
|
|
)
|
|
|
|
func init() {
|
|
indexAliasPattern = regexp.MustCompile(`\$(\d)`)
|
|
wildcardAliasPattern = regexp.MustCompile(`[*!]`)
|
|
}
|
|
|
|
type MQEResponse struct {
|
|
Success bool `json:"success"`
|
|
Name string `json:"name"`
|
|
Body []MQEResponseSerie `json:"body"`
|
|
}
|
|
|
|
type ResponseTimeRange struct {
|
|
Start int64 `json:"start"`
|
|
End int64 `json:"end"`
|
|
Resolution int64 `json:"Resolution"`
|
|
}
|
|
|
|
type MQEResponseSerie struct {
|
|
Query string `json:"query"`
|
|
Name string `json:"name"`
|
|
Type string `json:"type"`
|
|
Series []MQESerie `json:"series"`
|
|
TimeRange ResponseTimeRange `json:"timerange"`
|
|
}
|
|
|
|
type MQESerie struct {
|
|
Values []null.Float `json:"values"`
|
|
Tagset map[string]string `json:"tagset"`
|
|
}
|
|
|
|
type ResponseParser struct {
|
|
log log.Logger
|
|
}
|
|
|
|
func (parser *ResponseParser) Parse(res *http.Response, queryRef QueryToSend) ([]*tsdb.TimeSeries, error) {
|
|
body, err := ioutil.ReadAll(res.Body)
|
|
defer res.Body.Close()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if res.StatusCode/100 != 2 {
|
|
parser.log.Error("Request failed", "status code", res.StatusCode, "body", string(body))
|
|
return nil, fmt.Errorf("Returned invalid statuscode")
|
|
}
|
|
|
|
var data *MQEResponse = &MQEResponse{}
|
|
err = json.Unmarshal(body, data)
|
|
if err != nil {
|
|
parser.log.Info("Failed to unmarshal response", "error", err, "status", res.Status, "body", string(body))
|
|
return nil, err
|
|
}
|
|
|
|
if !data.Success {
|
|
return nil, fmt.Errorf("Request failed.")
|
|
}
|
|
|
|
var series []*tsdb.TimeSeries
|
|
for _, body := range data.Body {
|
|
for _, mqeSerie := range body.Series {
|
|
serie := &tsdb.TimeSeries{
|
|
Tags: map[string]string{},
|
|
Name: parser.formatLegend(body, mqeSerie, queryRef),
|
|
}
|
|
for key, value := range mqeSerie.Tagset {
|
|
serie.Tags[key] = value
|
|
}
|
|
|
|
for i, value := range mqeSerie.Values {
|
|
timestamp := body.TimeRange.Start + int64(i)*body.TimeRange.Resolution
|
|
serie.Points = append(serie.Points, tsdb.NewTimePoint(value, float64(timestamp)))
|
|
}
|
|
|
|
series = append(series, serie)
|
|
}
|
|
}
|
|
|
|
return series, nil
|
|
}
|
|
|
|
func (parser *ResponseParser) formatLegend(body MQEResponseSerie, mqeSerie MQESerie, queryToSend QueryToSend) string {
|
|
namePrefix := ""
|
|
|
|
//append predefined tags to seriename
|
|
for key, value := range mqeSerie.Tagset {
|
|
if key == "cluster" && queryToSend.QueryRef.AddClusterToAlias {
|
|
namePrefix += value + " "
|
|
}
|
|
}
|
|
for key, value := range mqeSerie.Tagset {
|
|
if key == "host" && queryToSend.QueryRef.AddHostToAlias {
|
|
namePrefix += value + " "
|
|
}
|
|
}
|
|
|
|
return namePrefix + parser.formatName(body, queryToSend)
|
|
}
|
|
|
|
func (parser *ResponseParser) formatName(body MQEResponseSerie, queryToSend QueryToSend) string {
|
|
if indexAliasPattern.MatchString(queryToSend.Metric.Alias) {
|
|
return parser.indexAlias(body, queryToSend)
|
|
}
|
|
|
|
if wildcardAliasPattern.MatchString(queryToSend.Metric.Metric) && wildcardAliasPattern.MatchString(queryToSend.Metric.Alias) {
|
|
return parser.wildcardAlias(body, queryToSend)
|
|
}
|
|
|
|
return body.Name
|
|
}
|
|
|
|
func (parser *ResponseParser) wildcardAlias(body MQEResponseSerie, queryToSend QueryToSend) string {
|
|
regString := strings.Replace(queryToSend.Metric.Metric, `*`, `(.*)`, 1)
|
|
reg, err := regexp.Compile(regString)
|
|
if err != nil {
|
|
return queryToSend.Metric.Alias
|
|
}
|
|
|
|
matches := reg.FindAllStringSubmatch(queryToSend.RawQuery, -1)
|
|
|
|
if len(matches) == 0 || len(matches[0]) < 2 {
|
|
return queryToSend.Metric.Alias
|
|
}
|
|
|
|
return matches[0][1]
|
|
}
|
|
|
|
func (parser *ResponseParser) indexAlias(body MQEResponseSerie, queryToSend QueryToSend) string {
|
|
queryNameParts := strings.Split(queryToSend.Metric.Metric, `.`)
|
|
|
|
name := indexAliasPattern.ReplaceAllStringFunc(queryToSend.Metric.Alias, func(in string) string {
|
|
positionName := strings.TrimSpace(strings.Replace(in, "$", "", 1))
|
|
|
|
pos, err := strconv.Atoi(positionName)
|
|
if err != nil {
|
|
return ""
|
|
}
|
|
|
|
for i, part := range queryNameParts {
|
|
if i == pos-1 {
|
|
return strings.TrimSpace(part)
|
|
}
|
|
}
|
|
|
|
return ""
|
|
})
|
|
|
|
return strings.Replace(name, " ", ".", -1)
|
|
}
|