query_test.go 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185
  1. package influxdb
  2. import (
  3. "testing"
  4. "strings"
  5. "github.com/grafana/grafana/pkg/tsdb"
  6. . "github.com/smartystreets/goconvey/convey"
  7. )
  8. func TestInfluxdbQueryBuilder(t *testing.T) {
  9. Convey("Influxdb query builder", t, func() {
  10. qp1, _ := NewQueryPart("field", []string{"value"})
  11. qp2, _ := NewQueryPart("mean", []string{})
  12. mathPartDivideBy100, _ := NewQueryPart("math", []string{"/ 100"})
  13. mathPartDivideByIntervalMs, _ := NewQueryPart("math", []string{"/ $__interval_ms"})
  14. groupBy1, _ := NewQueryPart("time", []string{"$__interval"})
  15. groupBy2, _ := NewQueryPart("tag", []string{"datacenter"})
  16. groupBy3, _ := NewQueryPart("fill", []string{"null"})
  17. groupByOldInterval, _ := NewQueryPart("time", []string{"$interval"})
  18. tag1 := &Tag{Key: "hostname", Value: "server1", Operator: "="}
  19. tag2 := &Tag{Key: "hostname", Value: "server2", Operator: "=", Condition: "OR"}
  20. queryContext := &tsdb.TsdbQuery{
  21. TimeRange: tsdb.NewTimeRange("5m", "now"),
  22. }
  23. Convey("can build simple query", func() {
  24. query := &Query{
  25. Selects: []*Select{{*qp1, *qp2}},
  26. Measurement: "cpu",
  27. Policy: "policy",
  28. GroupBy: []*QueryPart{groupBy1, groupBy3},
  29. Interval: "10s",
  30. }
  31. rawQuery, err := query.Build(queryContext)
  32. So(err, ShouldBeNil)
  33. So(rawQuery, ShouldEqual, `SELECT mean("value") FROM "policy"."cpu" WHERE time > now() - 5m GROUP BY time(10s) fill(null)`)
  34. })
  35. Convey("can build query with group bys", func() {
  36. query := &Query{
  37. Selects: []*Select{{*qp1, *qp2}},
  38. Measurement: "cpu",
  39. GroupBy: []*QueryPart{groupBy1, groupBy2, groupBy3},
  40. Tags: []*Tag{tag1, tag2},
  41. Interval: "5s",
  42. }
  43. rawQuery, err := query.Build(queryContext)
  44. So(err, ShouldBeNil)
  45. So(rawQuery, ShouldEqual, `SELECT mean("value") FROM "cpu" WHERE ("hostname" = 'server1' OR "hostname" = 'server2') AND time > now() - 5m GROUP BY time(5s), "datacenter" fill(null)`)
  46. })
  47. Convey("can build query with math part", func() {
  48. query := &Query{
  49. Selects: []*Select{{*qp1, *qp2, *mathPartDivideBy100}},
  50. Measurement: "cpu",
  51. Interval: "5s",
  52. }
  53. rawQuery, err := query.Build(queryContext)
  54. So(err, ShouldBeNil)
  55. So(rawQuery, ShouldEqual, `SELECT mean("value") / 100 FROM "cpu" WHERE time > now() - 5m`)
  56. })
  57. Convey("can build query with math part using $__interval_ms variable", func() {
  58. query := &Query{
  59. Selects: []*Select{{*qp1, *qp2, *mathPartDivideByIntervalMs}},
  60. Measurement: "cpu",
  61. Interval: "5s",
  62. }
  63. rawQuery, err := query.Build(queryContext)
  64. So(err, ShouldBeNil)
  65. So(rawQuery, ShouldEqual, `SELECT mean("value") / 5000 FROM "cpu" WHERE time > now() - 5m`)
  66. })
  67. Convey("can build query with old $interval variable", func() {
  68. query := &Query{
  69. Selects: []*Select{{*qp1, *qp2}},
  70. Measurement: "cpu",
  71. Policy: "",
  72. GroupBy: []*QueryPart{groupByOldInterval},
  73. }
  74. rawQuery, err := query.Build(queryContext)
  75. So(err, ShouldBeNil)
  76. So(rawQuery, ShouldEqual, `SELECT mean("value") FROM "cpu" WHERE time > now() - 5m GROUP BY time(200ms)`)
  77. })
  78. Convey("can render time range", func() {
  79. query := Query{}
  80. Convey("render from: 2h to now-1h", func() {
  81. query := Query{}
  82. queryContext := &tsdb.TsdbQuery{TimeRange: tsdb.NewTimeRange("2h", "now-1h")}
  83. So(query.renderTimeFilter(queryContext), ShouldEqual, "time > now() - 2h and time < now() - 1h")
  84. })
  85. Convey("render from: 10m", func() {
  86. queryContext := &tsdb.TsdbQuery{TimeRange: tsdb.NewTimeRange("10m", "now")}
  87. So(query.renderTimeFilter(queryContext), ShouldEqual, "time > now() - 10m")
  88. })
  89. })
  90. Convey("can build query from raw query", func() {
  91. query := &Query{
  92. Selects: []*Select{{*qp1, *qp2}},
  93. Measurement: "cpu",
  94. Policy: "policy",
  95. GroupBy: []*QueryPart{groupBy1, groupBy3},
  96. Interval: "10s",
  97. RawQuery: "Raw query",
  98. UseRawQuery: true,
  99. }
  100. rawQuery, err := query.Build(queryContext)
  101. So(err, ShouldBeNil)
  102. So(rawQuery, ShouldEqual, `Raw query`)
  103. })
  104. Convey("can render normal tags without operator", func() {
  105. query := &Query{Tags: []*Tag{{Operator: "", Value: `value`, Key: "key"}}}
  106. So(strings.Join(query.renderTags(), ""), ShouldEqual, `"key" = 'value'`)
  107. })
  108. Convey("can render regex tags without operator", func() {
  109. query := &Query{Tags: []*Tag{{Operator: "", Value: `/value/`, Key: "key"}}}
  110. So(strings.Join(query.renderTags(), ""), ShouldEqual, `"key" =~ /value/`)
  111. })
  112. Convey("can render regex tags", func() {
  113. query := &Query{Tags: []*Tag{{Operator: "=~", Value: `/value/`, Key: "key"}}}
  114. So(strings.Join(query.renderTags(), ""), ShouldEqual, `"key" =~ /value/`)
  115. })
  116. Convey("can render number tags", func() {
  117. query := &Query{Tags: []*Tag{{Operator: "=", Value: "10001", Key: "key"}}}
  118. So(strings.Join(query.renderTags(), ""), ShouldEqual, `"key" = '10001'`)
  119. })
  120. Convey("can render numbers less then condition tags", func() {
  121. query := &Query{Tags: []*Tag{{Operator: "<", Value: "10001", Key: "key"}}}
  122. So(strings.Join(query.renderTags(), ""), ShouldEqual, `"key" < 10001`)
  123. })
  124. Convey("can render number greather then condition tags", func() {
  125. query := &Query{Tags: []*Tag{{Operator: ">", Value: "10001", Key: "key"}}}
  126. So(strings.Join(query.renderTags(), ""), ShouldEqual, `"key" > 10001`)
  127. })
  128. Convey("can render string tags", func() {
  129. query := &Query{Tags: []*Tag{{Operator: "=", Value: "value", Key: "key"}}}
  130. So(strings.Join(query.renderTags(), ""), ShouldEqual, `"key" = 'value'`)
  131. })
  132. Convey("can render regular measurement", func() {
  133. query := &Query{Measurement: `apa`, Policy: "policy"}
  134. So(query.renderMeasurement(), ShouldEqual, ` FROM "policy"."apa"`)
  135. })
  136. Convey("can render regexp measurement", func() {
  137. query := &Query{Measurement: `/apa/`, Policy: "policy"}
  138. So(query.renderMeasurement(), ShouldEqual, ` FROM "policy"./apa/`)
  139. })
  140. })
  141. }