mirror of
https://github.com/grafana/grafana.git
synced 2024-11-26 02:40:26 -06:00
144 lines
3.9 KiB
Go
144 lines
3.9 KiB
Go
package cloudwatch
|
|
|
|
import (
|
|
"fmt"
|
|
"sort"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"github.com/aws/aws-sdk-go/aws"
|
|
"github.com/aws/aws-sdk-go/service/cloudwatch"
|
|
"github.com/grafana/grafana/pkg/services/featuremgmt"
|
|
)
|
|
|
|
func (e *cloudWatchExecutor) buildMetricDataQuery(query *cloudWatchQuery) (*cloudwatch.MetricDataQuery, error) {
|
|
mdq := &cloudwatch.MetricDataQuery{
|
|
Id: aws.String(query.Id),
|
|
ReturnData: aws.Bool(query.ReturnData),
|
|
}
|
|
|
|
if e.features.IsEnabled(featuremgmt.FlagCloudWatchDynamicLabels) && len(query.Label) > 0 {
|
|
mdq.Label = &query.Label
|
|
}
|
|
|
|
switch query.getGMDAPIMode() {
|
|
case GMDApiModeMathExpression:
|
|
mdq.Period = aws.Int64(int64(query.Period))
|
|
mdq.Expression = aws.String(query.Expression)
|
|
case GMDApiModeSQLExpression:
|
|
mdq.Period = aws.Int64(int64(query.Period))
|
|
mdq.Expression = aws.String(query.SqlExpression)
|
|
case GMDApiModeInferredSearchExpression:
|
|
mdq.Expression = aws.String(buildSearchExpression(query, query.Statistic))
|
|
case GMDApiModeMetricStat:
|
|
mdq.MetricStat = &cloudwatch.MetricStat{
|
|
Metric: &cloudwatch.Metric{
|
|
Namespace: aws.String(query.Namespace),
|
|
MetricName: aws.String(query.MetricName),
|
|
Dimensions: make([]*cloudwatch.Dimension, 0),
|
|
},
|
|
Period: aws.Int64(int64(query.Period)),
|
|
}
|
|
for key, values := range query.Dimensions {
|
|
mdq.MetricStat.Metric.Dimensions = append(mdq.MetricStat.Metric.Dimensions,
|
|
&cloudwatch.Dimension{
|
|
Name: aws.String(key),
|
|
Value: aws.String(values[0]),
|
|
})
|
|
}
|
|
mdq.MetricStat.Stat = aws.String(query.Statistic)
|
|
}
|
|
|
|
if mdq.Expression != nil {
|
|
query.UsedExpression = *mdq.Expression
|
|
} else {
|
|
query.UsedExpression = ""
|
|
}
|
|
|
|
return mdq, nil
|
|
}
|
|
|
|
func buildSearchExpression(query *cloudWatchQuery, stat string) string {
|
|
knownDimensions := make(map[string][]string)
|
|
dimensionNames := []string{}
|
|
dimensionNamesWithoutKnownValues := []string{}
|
|
|
|
for key, values := range query.Dimensions {
|
|
dimensionNames = append(dimensionNames, key)
|
|
hasWildcard := false
|
|
for _, value := range values {
|
|
if value == "*" {
|
|
hasWildcard = true
|
|
break
|
|
}
|
|
}
|
|
if hasWildcard {
|
|
dimensionNamesWithoutKnownValues = append(dimensionNamesWithoutKnownValues, key)
|
|
} else {
|
|
knownDimensions[key] = values
|
|
}
|
|
}
|
|
|
|
searchTerm := fmt.Sprintf(`MetricName="%s"`, query.MetricName)
|
|
keys := []string{}
|
|
for k := range knownDimensions {
|
|
keys = append(keys, k)
|
|
}
|
|
sort.Strings(keys)
|
|
for _, key := range keys {
|
|
values := escapeDoubleQuotes(knownDimensions[key])
|
|
valueExpression := join(values, " OR ", `"`, `"`)
|
|
if len(knownDimensions[key]) > 1 {
|
|
valueExpression = fmt.Sprintf(`(%s)`, valueExpression)
|
|
}
|
|
keyFilter := fmt.Sprintf(`"%s"=%s`, key, valueExpression)
|
|
searchTerm = appendSearch(searchTerm, keyFilter)
|
|
}
|
|
|
|
if query.MatchExact {
|
|
schema := fmt.Sprintf("%q", query.Namespace)
|
|
if len(dimensionNames) > 0 {
|
|
sort.Strings(dimensionNames)
|
|
schema += fmt.Sprintf(",%s", join(dimensionNames, ",", `"`, `"`))
|
|
}
|
|
return fmt.Sprintf("REMOVE_EMPTY(SEARCH('{%s} %s', '%s', %s))", schema, searchTerm, stat, strconv.Itoa(query.Period))
|
|
}
|
|
|
|
sort.Strings(dimensionNamesWithoutKnownValues)
|
|
searchTerm = appendSearch(searchTerm, join(dimensionNamesWithoutKnownValues, " ", `"`, `"`))
|
|
return fmt.Sprintf(`REMOVE_EMPTY(SEARCH('Namespace="%s" %s', '%s', %s))`, query.Namespace, searchTerm, stat, strconv.Itoa(query.Period))
|
|
}
|
|
|
|
func escapeDoubleQuotes(arr []string) []string {
|
|
result := []string{}
|
|
for _, value := range arr {
|
|
value = strings.ReplaceAll(value, `"`, `\"`)
|
|
result = append(result, value)
|
|
}
|
|
|
|
return result
|
|
}
|
|
|
|
func join(arr []string, delimiter string, valuePrefix string, valueSuffix string) string {
|
|
result := ""
|
|
for index, value := range arr {
|
|
result += valuePrefix + value + valueSuffix
|
|
if index+1 != len(arr) {
|
|
result += delimiter
|
|
}
|
|
}
|
|
|
|
return result
|
|
}
|
|
|
|
func appendSearch(target string, value string) string {
|
|
if value != "" {
|
|
if target == "" {
|
|
return value
|
|
}
|
|
return fmt.Sprintf("%v %v", target, value)
|
|
}
|
|
|
|
return target
|
|
}
|