2017-04-19 16:03:53 +00:00
|
|
|
package influx
|
|
|
|
|
|
|
|
import (
|
2017-06-06 22:26:08 +00:00
|
|
|
"encoding/json"
|
2017-06-16 22:54:15 +00:00
|
|
|
"reflect"
|
2017-04-19 16:03:53 +00:00
|
|
|
"testing"
|
2017-06-06 22:26:08 +00:00
|
|
|
"time"
|
2017-04-19 16:18:23 +00:00
|
|
|
|
|
|
|
"github.com/influxdata/chronograf"
|
2017-04-19 16:03:53 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func TestTemplateReplace(t *testing.T) {
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
query string
|
2017-06-06 22:26:08 +00:00
|
|
|
vars chronograf.TemplateVars
|
2017-04-19 16:03:53 +00:00
|
|
|
want string
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
name: "select with parameters",
|
|
|
|
query: "$METHOD field1, $field FROM $measurement WHERE temperature > $temperature",
|
2017-06-06 22:26:08 +00:00
|
|
|
vars: chronograf.TemplateVars{
|
|
|
|
chronograf.BasicTemplateVar{
|
2017-04-19 16:18:23 +00:00
|
|
|
Var: "$temperature",
|
2017-06-06 22:26:08 +00:00
|
|
|
Values: []chronograf.BasicTemplateValue{
|
2017-04-19 16:18:23 +00:00
|
|
|
{
|
|
|
|
Type: "csv",
|
|
|
|
Value: "10",
|
2017-04-19 16:03:53 +00:00
|
|
|
},
|
|
|
|
},
|
2017-04-19 16:18:23 +00:00
|
|
|
},
|
2017-06-06 22:26:08 +00:00
|
|
|
chronograf.BasicTemplateVar{
|
2017-04-19 16:18:23 +00:00
|
|
|
Var: "$field",
|
2017-06-06 22:26:08 +00:00
|
|
|
Values: []chronograf.BasicTemplateValue{
|
2017-04-19 16:18:23 +00:00
|
|
|
{
|
|
|
|
Type: "fieldKey",
|
|
|
|
Value: "field2",
|
2017-04-19 16:03:53 +00:00
|
|
|
},
|
|
|
|
},
|
2017-04-19 16:18:23 +00:00
|
|
|
},
|
2017-06-06 22:26:08 +00:00
|
|
|
chronograf.BasicTemplateVar{
|
2017-04-19 16:18:23 +00:00
|
|
|
Var: "$METHOD",
|
2017-06-06 22:26:08 +00:00
|
|
|
Values: []chronograf.BasicTemplateValue{
|
2017-04-19 16:18:23 +00:00
|
|
|
{
|
|
|
|
Type: "csv",
|
|
|
|
Value: "SELECT",
|
2017-04-19 16:03:53 +00:00
|
|
|
},
|
|
|
|
},
|
2017-04-19 16:18:23 +00:00
|
|
|
},
|
2017-06-06 22:26:08 +00:00
|
|
|
chronograf.BasicTemplateVar{
|
2017-04-19 16:18:23 +00:00
|
|
|
Var: "$measurement",
|
2017-06-06 22:26:08 +00:00
|
|
|
Values: []chronograf.BasicTemplateValue{
|
2017-04-19 16:18:23 +00:00
|
|
|
{
|
|
|
|
Type: "csv",
|
|
|
|
Value: `"cpu"`,
|
2017-04-19 16:03:53 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
want: `SELECT field1, "field2" FROM "cpu" WHERE temperature > 10`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "select with parameters and aggregates",
|
|
|
|
query: `SELECT mean($field) FROM "cpu" WHERE $tag = $value GROUP BY $tag`,
|
2017-06-06 22:26:08 +00:00
|
|
|
vars: chronograf.TemplateVars{
|
|
|
|
chronograf.BasicTemplateVar{
|
2017-04-19 16:18:23 +00:00
|
|
|
Var: "$value",
|
2017-06-06 22:26:08 +00:00
|
|
|
Values: []chronograf.BasicTemplateValue{
|
2017-04-19 16:18:23 +00:00
|
|
|
{
|
|
|
|
Type: "tagValue",
|
|
|
|
Value: "howdy.com",
|
2017-04-19 16:03:53 +00:00
|
|
|
},
|
|
|
|
},
|
2017-04-19 16:18:23 +00:00
|
|
|
},
|
2017-06-06 22:26:08 +00:00
|
|
|
chronograf.BasicTemplateVar{
|
2017-04-19 16:18:23 +00:00
|
|
|
Var: "$tag",
|
2017-06-06 22:26:08 +00:00
|
|
|
Values: []chronograf.BasicTemplateValue{
|
2017-04-19 16:18:23 +00:00
|
|
|
{
|
|
|
|
Type: "tagKey",
|
|
|
|
Value: "host",
|
2017-04-19 16:03:53 +00:00
|
|
|
},
|
|
|
|
},
|
2017-04-19 16:18:23 +00:00
|
|
|
},
|
2017-06-06 22:26:08 +00:00
|
|
|
chronograf.BasicTemplateVar{
|
2017-04-19 16:18:23 +00:00
|
|
|
Var: "$field",
|
2017-06-06 22:26:08 +00:00
|
|
|
Values: []chronograf.BasicTemplateValue{
|
2017-04-19 16:18:23 +00:00
|
|
|
{
|
|
|
|
Type: "fieldKey",
|
|
|
|
Value: "field",
|
2017-04-19 16:03:53 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
want: `SELECT mean("field") FROM "cpu" WHERE "host" = 'howdy.com' GROUP BY "host"`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "Non-existant parameters",
|
|
|
|
query: `SELECT $field FROM "cpu"`,
|
|
|
|
want: `SELECT $field FROM "cpu"`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "var without a value",
|
|
|
|
query: `SELECT $field FROM "cpu"`,
|
2017-06-06 22:26:08 +00:00
|
|
|
vars: chronograf.TemplateVars{
|
|
|
|
chronograf.BasicTemplateVar{
|
2017-04-19 16:18:23 +00:00
|
|
|
Var: "$field",
|
2017-04-19 16:03:53 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
want: `SELECT $field FROM "cpu"`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "var with unknown type",
|
|
|
|
query: `SELECT $field FROM "cpu"`,
|
2017-06-06 22:26:08 +00:00
|
|
|
vars: chronograf.TemplateVars{
|
|
|
|
chronograf.BasicTemplateVar{
|
2017-04-19 16:18:23 +00:00
|
|
|
Var: "$field",
|
2017-06-06 22:26:08 +00:00
|
|
|
Values: []chronograf.BasicTemplateValue{
|
2017-04-19 16:18:23 +00:00
|
|
|
{
|
|
|
|
Type: "who knows?",
|
|
|
|
Value: "field",
|
2017-04-19 16:03:53 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
want: `SELECT $field FROM "cpu"`,
|
|
|
|
},
|
2017-06-06 22:26:08 +00:00
|
|
|
{
|
|
|
|
name: "auto group by",
|
2017-06-15 20:17:19 +00:00
|
|
|
query: `SELECT mean(usage_idle) from "cpu" where time > now() - 4320h group by :interval:`,
|
2017-06-06 22:26:08 +00:00
|
|
|
vars: chronograf.TemplateVars{
|
|
|
|
&chronograf.GroupByVar{
|
2017-06-15 20:17:19 +00:00
|
|
|
Var: ":interval:",
|
2017-06-06 22:26:08 +00:00
|
|
|
Duration: 180 * 24 * time.Hour,
|
|
|
|
Resolution: 1000,
|
|
|
|
ReportingInterval: 10 * time.Second,
|
|
|
|
},
|
|
|
|
},
|
2017-10-03 23:31:51 +00:00
|
|
|
want: `SELECT mean(usage_idle) from "cpu" where time > now() - 4320h group by time(46656s)`,
|
2017-06-08 15:31:35 +00:00
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "auto group by without duration",
|
2017-06-15 20:17:19 +00:00
|
|
|
query: `SELECT mean(usage_idle) from "cpu" WHERE time > now() - 4320h group by :interval:`,
|
2017-06-08 15:31:35 +00:00
|
|
|
vars: chronograf.TemplateVars{
|
|
|
|
&chronograf.GroupByVar{
|
2017-06-15 20:17:19 +00:00
|
|
|
Var: ":interval:",
|
2017-06-08 15:31:35 +00:00
|
|
|
Duration: 0 * time.Minute,
|
|
|
|
Resolution: 1000,
|
|
|
|
ReportingInterval: 10 * time.Second,
|
|
|
|
},
|
|
|
|
},
|
2017-10-03 23:31:51 +00:00
|
|
|
want: `SELECT mean(usage_idle) from "cpu" WHERE time > now() - 4320h group by time(46656s)`,
|
2017-06-13 18:42:52 +00:00
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "auto group by with :dashboardTime:",
|
2017-06-15 20:17:19 +00:00
|
|
|
query: `SELECT mean(usage_idle) from "cpu" WHERE time > :dashboardTime: group by :interval:`,
|
2017-06-13 18:42:52 +00:00
|
|
|
vars: chronograf.TemplateVars{
|
|
|
|
&chronograf.GroupByVar{
|
2017-06-15 20:17:19 +00:00
|
|
|
Var: ":interval:",
|
2017-06-13 18:42:52 +00:00
|
|
|
Duration: 0 * time.Minute,
|
|
|
|
Resolution: 1000,
|
|
|
|
ReportingInterval: 10 * time.Second,
|
|
|
|
},
|
|
|
|
&chronograf.BasicTemplateVar{
|
|
|
|
Var: ":dashboardTime:",
|
|
|
|
Values: []chronograf.BasicTemplateValue{
|
|
|
|
{
|
|
|
|
Type: "constant",
|
|
|
|
Value: "now() - 4320h",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2017-10-03 23:31:51 +00:00
|
|
|
want: `SELECT mean(usage_idle) from "cpu" WHERE time > now() - 4320h group by time(46656s)`,
|
2017-06-06 22:26:08 +00:00
|
|
|
},
|
2017-06-16 22:54:15 +00:00
|
|
|
{
|
|
|
|
name: "auto group by failing condition",
|
|
|
|
query: `SELECT mean(usage_idle) FROM "cpu" WHERE time > :dashboardTime: GROUP BY :interval:`,
|
|
|
|
vars: []chronograf.TemplateVariable{
|
|
|
|
&chronograf.GroupByVar{
|
|
|
|
Var: ":interval:",
|
|
|
|
Resolution: 115,
|
|
|
|
ReportingInterval: 10 * time.Second,
|
|
|
|
},
|
|
|
|
chronograf.BasicTemplateVar{
|
|
|
|
Var: ":dashboardTime:",
|
|
|
|
Values: []chronograf.BasicTemplateValue{
|
|
|
|
{
|
|
|
|
Value: "now() - 1h",
|
|
|
|
Type: "constant",
|
|
|
|
Selected: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2017-10-03 23:31:51 +00:00
|
|
|
want: `SELECT mean(usage_idle) FROM "cpu" WHERE time > now() - 1h GROUP BY time(93s)`,
|
2017-06-16 22:54:15 +00:00
|
|
|
},
|
2017-04-19 16:03:53 +00:00
|
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
|
|
got := TemplateReplace(tt.query, tt.vars)
|
|
|
|
if got != tt.want {
|
|
|
|
t.Errorf("TestParse %s =\n%s\nwant\n%s", tt.name, got, tt.want)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
2017-06-06 22:26:08 +00:00
|
|
|
|
|
|
|
func Test_TemplateVarsUnmarshalling(t *testing.T) {
|
|
|
|
req := `[
|
|
|
|
{
|
2017-06-15 20:17:19 +00:00
|
|
|
"tempVar": ":interval:",
|
2017-06-06 22:26:08 +00:00
|
|
|
"resolution": 1000,
|
|
|
|
"reportingInterval": 10
|
|
|
|
},
|
|
|
|
{
|
2017-06-16 22:54:15 +00:00
|
|
|
"tempVar": ":cpu:",
|
2017-06-06 22:26:08 +00:00
|
|
|
"values": [
|
|
|
|
{
|
|
|
|
"type": "tagValue",
|
2017-06-07 15:16:09 +00:00
|
|
|
"value": "cpu-total",
|
|
|
|
"selected": false
|
2017-06-06 22:26:08 +00:00
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]`
|
|
|
|
|
2017-06-16 22:54:15 +00:00
|
|
|
expected := []chronograf.TemplateVariable{
|
|
|
|
&chronograf.GroupByVar{
|
|
|
|
Var: ":interval:",
|
|
|
|
Resolution: 1000,
|
|
|
|
ReportingInterval: 10 * time.Nanosecond,
|
|
|
|
},
|
|
|
|
chronograf.BasicTemplateVar{
|
|
|
|
Var: ":cpu:",
|
|
|
|
Values: []chronograf.BasicTemplateValue{
|
|
|
|
{
|
|
|
|
Value: "cpu-total",
|
|
|
|
Type: "tagValue",
|
|
|
|
Selected: false,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2017-06-06 22:26:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
var tvars chronograf.TemplateVars
|
|
|
|
err := json.Unmarshal([]byte(req), &tvars)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal("Err unmarshaling:", err)
|
|
|
|
}
|
|
|
|
|
2017-06-07 15:16:09 +00:00
|
|
|
if len(tvars) != len(expected) {
|
|
|
|
t.Fatal("Expected", len(expected), "vars but found", len(tvars))
|
|
|
|
}
|
|
|
|
|
2017-06-16 22:54:15 +00:00
|
|
|
if !reflect.DeepEqual(*(tvars[0].(*chronograf.GroupByVar)), *(expected[0].(*chronograf.GroupByVar))) {
|
|
|
|
t.Errorf("UnmarshalJSON() = \n%#v\n want \n%#v\n", *(tvars[0].(*chronograf.GroupByVar)), *(expected[0].(*chronograf.GroupByVar)))
|
|
|
|
}
|
|
|
|
|
|
|
|
if !reflect.DeepEqual(tvars[1].(chronograf.BasicTemplateVar), expected[1].(chronograf.BasicTemplateVar)) {
|
|
|
|
t.Errorf("UnmarshalJSON() = \n%#v\n want \n%#v\n", tvars[1].(chronograf.BasicTemplateVar), expected[1].(chronograf.BasicTemplateVar))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestGroupByVarString(t *testing.T) {
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
tvar *chronograf.GroupByVar
|
|
|
|
want string
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
name: "String() calculates the GROUP BY interval",
|
|
|
|
tvar: &chronograf.GroupByVar{
|
|
|
|
Resolution: 700,
|
|
|
|
ReportingInterval: 10 * time.Second,
|
|
|
|
Duration: 24 * time.Hour,
|
|
|
|
},
|
2017-10-11 20:14:44 +00:00
|
|
|
want: "time(370s)",
|
2017-06-16 22:54:15 +00:00
|
|
|
},
|
|
|
|
{
|
2017-10-16 21:15:12 +00:00
|
|
|
name: "String() milliseconds if less than one second intervals",
|
2017-06-16 22:54:15 +00:00
|
|
|
tvar: &chronograf.GroupByVar{
|
|
|
|
Resolution: 100000,
|
|
|
|
ReportingInterval: 10 * time.Second,
|
|
|
|
Duration: time.Hour,
|
|
|
|
},
|
2017-10-11 20:14:44 +00:00
|
|
|
want: "time(107ms)",
|
2017-06-16 22:54:15 +00:00
|
|
|
},
|
|
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
|
|
got := tt.tvar.String()
|
|
|
|
if got != tt.want {
|
|
|
|
t.Errorf("TestGroupByVarString %s =\n%s\nwant\n%s", tt.name, got, tt.want)
|
|
|
|
}
|
|
|
|
})
|
2017-06-06 22:26:08 +00:00
|
|
|
}
|
|
|
|
}
|