query_test.go 6.5 KB

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