annotation_query.go 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120
  1. package stackdriver
  2. import (
  3. "context"
  4. "strconv"
  5. "strings"
  6. "time"
  7. "github.com/grafana/grafana/pkg/tsdb"
  8. )
  9. func (e *StackdriverExecutor) executeAnnotationQuery(ctx context.Context, tsdbQuery *tsdb.TsdbQuery) (*tsdb.Response, error) {
  10. result := &tsdb.Response{
  11. Results: make(map[string]*tsdb.QueryResult),
  12. }
  13. firstQuery := tsdbQuery.Queries[0]
  14. queries, err := e.buildQueries(tsdbQuery)
  15. if err != nil {
  16. return nil, err
  17. }
  18. queryRes, resp, err := e.executeQuery(ctx, queries[0], tsdbQuery)
  19. if err != nil {
  20. return nil, err
  21. }
  22. title := firstQuery.Model.Get("title").MustString()
  23. text := firstQuery.Model.Get("text").MustString()
  24. tags := firstQuery.Model.Get("tags").MustString()
  25. err = e.parseToAnnotations(queryRes, resp, queries[0], title, text, tags)
  26. result.Results[firstQuery.RefId] = queryRes
  27. return result, err
  28. }
  29. func (e *StackdriverExecutor) parseToAnnotations(queryRes *tsdb.QueryResult, data StackdriverResponse, query *StackdriverQuery, title string, text string, tags string) error {
  30. annotations := make([]map[string]string, 0)
  31. for _, series := range data.TimeSeries {
  32. // reverse the order to be ascending
  33. for i := len(series.Points) - 1; i >= 0; i-- {
  34. point := series.Points[i]
  35. value := strconv.FormatFloat(point.Value.DoubleValue, 'f', 6, 64)
  36. if series.ValueType == "STRING" {
  37. value = point.Value.StringValue
  38. }
  39. annotation := make(map[string]string)
  40. annotation["time"] = point.Interval.EndTime.UTC().Format(time.RFC3339)
  41. annotation["title"] = formatAnnotationText(title, value, series.Metric.Type, series.Metric.Labels, series.Resource.Labels)
  42. annotation["tags"] = tags
  43. annotation["text"] = formatAnnotationText(text, value, series.Metric.Type, series.Metric.Labels, series.Resource.Labels)
  44. annotations = append(annotations, annotation)
  45. }
  46. }
  47. transformAnnotationToTable(annotations, queryRes)
  48. return nil
  49. }
  50. func transformAnnotationToTable(data []map[string]string, result *tsdb.QueryResult) {
  51. table := &tsdb.Table{
  52. Columns: make([]tsdb.TableColumn, 4),
  53. Rows: make([]tsdb.RowValues, 0),
  54. }
  55. table.Columns[0].Text = "time"
  56. table.Columns[1].Text = "title"
  57. table.Columns[2].Text = "tags"
  58. table.Columns[3].Text = "text"
  59. for _, r := range data {
  60. values := make([]interface{}, 4)
  61. values[0] = r["time"]
  62. values[1] = r["title"]
  63. values[2] = r["tags"]
  64. values[3] = r["text"]
  65. table.Rows = append(table.Rows, values)
  66. }
  67. result.Tables = append(result.Tables, table)
  68. result.Meta.Set("rowCount", len(data))
  69. slog.Info("anno", "len", len(data))
  70. }
  71. func formatAnnotationText(annotationText string, pointValue string, metricType string, metricLabels map[string]string, resourceLabels map[string]string) string {
  72. result := legendKeyFormat.ReplaceAllFunc([]byte(annotationText), func(in []byte) []byte {
  73. metaPartName := strings.Replace(string(in), "{{", "", 1)
  74. metaPartName = strings.Replace(metaPartName, "}}", "", 1)
  75. metaPartName = strings.TrimSpace(metaPartName)
  76. if metaPartName == "metric.type" {
  77. return []byte(metricType)
  78. }
  79. metricPart := replaceWithMetricPart(metaPartName, metricType)
  80. if metricPart != nil {
  81. return metricPart
  82. }
  83. if metaPartName == "metric.value" {
  84. return []byte(pointValue)
  85. }
  86. metaPartName = strings.Replace(metaPartName, "metric.label.", "", 1)
  87. if val, exists := metricLabels[metaPartName]; exists {
  88. return []byte(val)
  89. }
  90. metaPartName = strings.Replace(metaPartName, "resource.label.", "", 1)
  91. if val, exists := resourceLabels[metaPartName]; exists {
  92. return []byte(val)
  93. }
  94. return in
  95. })
  96. return string(result)
  97. }