2019-07-07 16:44:26 +00:00
|
|
|
package prometheus
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2019-08-20 19:52:49 +00:00
|
|
|
"math"
|
2019-07-07 16:44:26 +00:00
|
|
|
"net/http"
|
|
|
|
"net/url"
|
2019-08-20 19:52:49 +00:00
|
|
|
"reflect"
|
2019-07-07 20:34:04 +00:00
|
|
|
"testing"
|
2019-08-20 19:52:49 +00:00
|
|
|
|
|
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/app/vmselect/netstorage"
|
2019-07-07 16:44:26 +00:00
|
|
|
)
|
|
|
|
|
2020-07-20 12:28:36 +00:00
|
|
|
func TestRemoveEmptyValuesAndTimeseries(t *testing.T) {
|
2019-08-20 19:52:49 +00:00
|
|
|
f := func(tss []netstorage.Result, tssExpected []netstorage.Result) {
|
|
|
|
t.Helper()
|
2020-07-20 12:28:36 +00:00
|
|
|
tss = removeEmptyValuesAndTimeseries(tss)
|
2019-08-20 19:52:49 +00:00
|
|
|
if !reflect.DeepEqual(tss, tssExpected) {
|
|
|
|
t.Fatalf("unexpected result; got %v; want %v", tss, tssExpected)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
f(nil, nil)
|
|
|
|
f([]netstorage.Result{
|
|
|
|
{
|
|
|
|
Timestamps: []int64{100, 200, 300},
|
|
|
|
Values: []float64{1, 2, 3},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Timestamps: []int64{100, 200, 300, 400},
|
|
|
|
Values: []float64{nan, nan, 3, nan},
|
|
|
|
},
|
2020-07-20 12:28:36 +00:00
|
|
|
{
|
|
|
|
Timestamps: []int64{1, 2},
|
|
|
|
Values: []float64{nan, nan},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Timestamps: nil,
|
|
|
|
Values: nil,
|
|
|
|
},
|
2019-08-20 19:52:49 +00:00
|
|
|
}, []netstorage.Result{
|
|
|
|
{
|
|
|
|
Timestamps: []int64{100, 200, 300},
|
|
|
|
Values: []float64{1, 2, 3},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Timestamps: []int64{300},
|
|
|
|
Values: []float64{3},
|
|
|
|
},
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-07-07 16:44:26 +00:00
|
|
|
func TestGetTimeSuccess(t *testing.T) {
|
|
|
|
f := func(s string, timestampExpected int64) {
|
|
|
|
t.Helper()
|
|
|
|
urlStr := fmt.Sprintf("http://foo.bar/baz?s=%s", url.QueryEscape(s))
|
|
|
|
r, err := http.NewRequest("GET", urlStr, nil)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("unexpected error in NewRequest: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Verify defaultValue
|
|
|
|
ts, err := getTime(r, "foo", 123)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("unexpected error when obtaining default time from getTime(%q): %s", s, err)
|
|
|
|
}
|
|
|
|
if ts != 123 {
|
|
|
|
t.Fatalf("unexpected default value for getTime(%q); got %d; want %d", s, ts, 123)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Verify timestampExpected
|
|
|
|
ts, err = getTime(r, "s", 123)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("unexpected error in getTime(%q): %s", s, err)
|
|
|
|
}
|
|
|
|
if ts != timestampExpected {
|
|
|
|
t.Fatalf("unexpected timestamp for getTime(%q); got %d; want %d", s, ts, timestampExpected)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
f("2019-07-07T20:01:02Z", 1562529662000)
|
|
|
|
f("2019-07-07T20:47:40+03:00", 1562521660000)
|
2019-07-11 14:07:20 +00:00
|
|
|
f("-292273086-05-16T16:47:06Z", minTimeMsecs)
|
|
|
|
f("292277025-08-18T07:12:54.999999999Z", maxTimeMsecs)
|
2019-07-07 16:44:26 +00:00
|
|
|
f("1562529662.324", 1562529662324)
|
2019-07-11 14:07:20 +00:00
|
|
|
f("-9223372036.854", minTimeMsecs)
|
|
|
|
f("-9223372036.855", minTimeMsecs)
|
|
|
|
f("9223372036.855", maxTimeMsecs)
|
2019-07-07 16:44:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestGetTimeError(t *testing.T) {
|
|
|
|
f := func(s string) {
|
|
|
|
t.Helper()
|
|
|
|
urlStr := fmt.Sprintf("http://foo.bar/baz?s=%s", url.QueryEscape(s))
|
|
|
|
r, err := http.NewRequest("GET", urlStr, nil)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("unexpected error in NewRequest: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Verify defaultValue
|
|
|
|
ts, err := getTime(r, "foo", 123)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("unexpected error when obtaining default time from getTime(%q): %s", s, err)
|
|
|
|
}
|
|
|
|
if ts != 123 {
|
|
|
|
t.Fatalf("unexpected default value for getTime(%q); got %d; want %d", s, ts, 123)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Verify timestampExpected
|
2019-07-07 20:34:04 +00:00
|
|
|
_, err = getTime(r, "s", 123)
|
2019-07-07 16:44:26 +00:00
|
|
|
if err == nil {
|
|
|
|
t.Fatalf("expecting non-nil error in getTime(%q)", s)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
f("foo")
|
|
|
|
f("2019-07-07T20:01:02Zisdf")
|
|
|
|
f("2019-07-07T20:47:40+03:00123")
|
|
|
|
f("-292273086-05-16T16:47:07Z")
|
|
|
|
f("292277025-08-18T07:12:54.999999998Z")
|
|
|
|
}
|
2020-07-05 14:56:54 +00:00
|
|
|
|
|
|
|
func TestAdjustLastPoints(t *testing.T) {
|
2020-07-05 15:17:02 +00:00
|
|
|
f := func(tss []netstorage.Result, start, end int64, tssExpected []netstorage.Result) {
|
2020-07-05 14:56:54 +00:00
|
|
|
t.Helper()
|
2020-07-05 15:17:02 +00:00
|
|
|
tss = adjustLastPoints(tss, start, end)
|
2020-07-05 14:56:54 +00:00
|
|
|
for i, ts := range tss {
|
|
|
|
for j, value := range ts.Values {
|
|
|
|
expectedValue := tssExpected[i].Values[j]
|
|
|
|
if math.IsNaN(expectedValue) {
|
|
|
|
if !math.IsNaN(value) {
|
2020-07-14 09:45:42 +00:00
|
|
|
t.Fatalf("unexpected value for time series #%d at position %d; got %v; want nan", i, j, value)
|
2020-07-05 14:56:54 +00:00
|
|
|
}
|
|
|
|
} else if expectedValue != value {
|
2020-07-14 09:45:42 +00:00
|
|
|
t.Fatalf("unexpected value for time series #%d at position %d; got %v; want %v", i, j, value, expectedValue)
|
2020-07-05 14:56:54 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if !reflect.DeepEqual(ts.Timestamps, tssExpected[i].Timestamps) {
|
2020-07-14 09:45:42 +00:00
|
|
|
t.Fatalf("unexpected timestamps for time series #%d; got %v; want %v", i, tss, tssExpected)
|
2020-07-05 14:56:54 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
nan := math.NaN()
|
|
|
|
|
2020-07-05 15:17:02 +00:00
|
|
|
f(nil, 300, 500, nil)
|
2020-07-05 14:56:54 +00:00
|
|
|
|
|
|
|
f([]netstorage.Result{
|
|
|
|
{
|
|
|
|
Timestamps: []int64{100, 200, 300, 400, 500},
|
|
|
|
Values: []float64{1, 2, 3, 4, nan},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Timestamps: []int64{100, 200, 300, 400, 500},
|
|
|
|
Values: []float64{1, 2, 3, nan, nan},
|
|
|
|
},
|
2020-07-05 15:17:02 +00:00
|
|
|
}, 400, 500, []netstorage.Result{
|
2020-07-05 14:56:54 +00:00
|
|
|
{
|
|
|
|
Timestamps: []int64{100, 200, 300, 400, 500},
|
|
|
|
Values: []float64{1, 2, 3, 4, 4},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Timestamps: []int64{100, 200, 300, 400, 500},
|
|
|
|
Values: []float64{1, 2, 3, nan, nan},
|
|
|
|
},
|
|
|
|
})
|
|
|
|
|
|
|
|
f([]netstorage.Result{
|
|
|
|
{
|
|
|
|
Timestamps: []int64{100, 200, 300, 400, 500},
|
|
|
|
Values: []float64{1, 2, 3, nan, nan},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Timestamps: []int64{100, 200, 300, 400, 500},
|
|
|
|
Values: []float64{1, 2, nan, nan, nan},
|
|
|
|
},
|
2020-07-05 15:17:02 +00:00
|
|
|
}, 300, 500, []netstorage.Result{
|
2020-07-05 14:56:54 +00:00
|
|
|
{
|
|
|
|
Timestamps: []int64{100, 200, 300, 400, 500},
|
|
|
|
Values: []float64{1, 2, 3, 3, 3},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Timestamps: []int64{100, 200, 300, 400, 500},
|
|
|
|
Values: []float64{1, 2, nan, nan, nan},
|
|
|
|
},
|
|
|
|
})
|
|
|
|
|
|
|
|
f([]netstorage.Result{
|
|
|
|
{
|
|
|
|
Timestamps: []int64{100, 200, 300, 400, 500},
|
|
|
|
Values: []float64{1, 2, nan, nan, nan},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Timestamps: []int64{100, 200, 300, 400, 500},
|
|
|
|
Values: []float64{1, nan, nan, nan, nan},
|
|
|
|
},
|
|
|
|
}, 500, 300, []netstorage.Result{
|
|
|
|
{
|
|
|
|
Timestamps: []int64{100, 200, 300, 400, 500},
|
|
|
|
Values: []float64{1, 2, nan, nan, nan},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Timestamps: []int64{100, 200, 300, 400, 500},
|
|
|
|
Values: []float64{1, nan, nan, nan, nan},
|
|
|
|
},
|
|
|
|
})
|
|
|
|
|
|
|
|
f([]netstorage.Result{
|
|
|
|
{
|
|
|
|
Timestamps: []int64{100, 200, 300, 400, 500},
|
|
|
|
Values: []float64{1, 2, 3, 4, nan},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Timestamps: []int64{100, 200, 300, 400},
|
|
|
|
Values: []float64{1, 2, 3, 4},
|
|
|
|
},
|
2020-07-05 15:17:02 +00:00
|
|
|
}, 400, 500, []netstorage.Result{
|
2020-07-05 14:56:54 +00:00
|
|
|
{
|
|
|
|
Timestamps: []int64{100, 200, 300, 400, 500},
|
|
|
|
Values: []float64{1, 2, 3, 4, 4},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Timestamps: []int64{100, 200, 300, 400},
|
|
|
|
Values: []float64{1, 2, 3, 4},
|
|
|
|
},
|
|
|
|
})
|
|
|
|
|
|
|
|
f([]netstorage.Result{
|
|
|
|
{
|
|
|
|
Timestamps: []int64{100, 200, 300, 400, 500},
|
|
|
|
Values: []float64{1, 2, 3, nan, nan},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Timestamps: []int64{100, 200, 300},
|
|
|
|
Values: []float64{1, 2, nan},
|
|
|
|
},
|
2020-07-05 15:17:02 +00:00
|
|
|
}, 300, 600, []netstorage.Result{
|
2020-07-05 14:56:54 +00:00
|
|
|
{
|
|
|
|
Timestamps: []int64{100, 200, 300, 400, 500},
|
|
|
|
Values: []float64{1, 2, 3, 3, 3},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Timestamps: []int64{100, 200, 300},
|
|
|
|
Values: []float64{1, 2, nan},
|
|
|
|
},
|
|
|
|
})
|
2020-07-14 09:45:42 +00:00
|
|
|
|
|
|
|
// Check for timestamps outside the configured time range.
|
|
|
|
// See https://github.com/VictoriaMetrics/VictoriaMetrics/issues/625
|
|
|
|
f([]netstorage.Result{
|
|
|
|
{
|
|
|
|
Timestamps: []int64{100, 200, 300, 400, 500},
|
|
|
|
Values: []float64{1, 2, 3, nan, nan},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Timestamps: []int64{100, 200, 300},
|
|
|
|
Values: []float64{1, 2, 45},
|
|
|
|
},
|
|
|
|
}, 250, 400, []netstorage.Result{
|
|
|
|
{
|
|
|
|
Timestamps: []int64{100, 200, 300, 400, 500},
|
|
|
|
Values: []float64{1, 2, 3, nan, nan},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Timestamps: []int64{100, 200, 300},
|
|
|
|
Values: []float64{1, 2, 2},
|
|
|
|
},
|
|
|
|
})
|
2020-07-05 14:56:54 +00:00
|
|
|
}
|