app/vmselect/promql: properly handle scalar default vector, scalar if vector and scalar ifnot vector queries

Previously `vector` time series could be unexpectedly returned from such queries
This commit is contained in:
Aliaksandr Valialkin 2022-04-21 15:33:27 +03:00
parent bc54ae9608
commit dac24aa342
No known key found for this signature in database
GPG key ID: A72BEC6CD3D0DED1
3 changed files with 112 additions and 6 deletions

View file

@ -136,7 +136,7 @@ func newBinaryOpFunc(bf func(left, right float64, isBool bool) float64) binaryOp
func adjustBinaryOpTags(be *metricsql.BinaryOpExpr, left, right []*timeseries) ([]*timeseries, []*timeseries, []*timeseries, error) {
if len(be.GroupModifier.Op) == 0 && len(be.JoinModifier.Op) == 0 {
if isScalar(left) {
if isScalar(left) && be.Op != "default" && be.Op != "if" && be.Op != "ifnot" {
// Fast path: `scalar op vector`
rvsLeft := make([]*timeseries, len(right))
tsLeft := left[0]

View file

@ -1855,10 +1855,10 @@ func TestExecSuccess(t *testing.T) {
})
t.Run(`drop_common_labels(multi_series)`, func(t *testing.T) {
t.Parallel()
q := `drop_common_labels((
q := `sort_desc(drop_common_labels((
label_set(time(), "foo", "bar", "__name__", "xxx", "q", "we"),
label_set(time()/10, "foo", "bar", "__name__", "yyy"),
))`
)))`
r1 := netstorage.Result{
MetricName: metricNameExpected,
Values: []float64{1000, 1200, 1400, 1600, 1800, 2000},
@ -1880,10 +1880,10 @@ func TestExecSuccess(t *testing.T) {
})
t.Run(`drop_common_labels(multi_args)`, func(t *testing.T) {
t.Parallel()
q := `drop_common_labels(
q := `sort(drop_common_labels(
label_set(time(), "foo", "bar", "__name__", "xxx", "q", "we"),
label_set(time()/10, "foo", "bar", "__name__", "xxx"),
)`
))`
r1 := netstorage.Result{
MetricName: metricNameExpected,
Values: []float64{100, 120, 140, 160, 180, 200},
@ -2640,7 +2640,7 @@ func TestExecSuccess(t *testing.T) {
q := `1 != nan`
r := netstorage.Result{
MetricName: metricNameExpected,
Values: []float64{1, 1, 1, 1, 1, 1},
Values: []float64{1, 1, 1, 1, 1, 1},
Timestamps: timestampsExpected,
}
resultExpected := []netstorage.Result{r}
@ -2761,6 +2761,37 @@ func TestExecSuccess(t *testing.T) {
resultExpected := []netstorage.Result{r}
f(q, resultExpected)
})
t.Run(`scalar default scalar_from_vector`, func(t *testing.T) {
t.Parallel()
q := `time() > 1400 default scalar(label_set(123, "foo", "bar"))`
r := netstorage.Result{
MetricName: metricNameExpected,
Values: []float64{123, 123, 123, 1600, 1800, 2000},
Timestamps: timestampsExpected,
}
resultExpected := []netstorage.Result{r}
f(q, resultExpected)
})
t.Run(`scalar default vector1`, func(t *testing.T) {
t.Parallel()
q := `time() > 1400 default label_set(123, "foo", "bar")`
resultExpected := []netstorage.Result{}
f(q, resultExpected)
})
t.Run(`scalar default vector2`, func(t *testing.T) {
t.Parallel()
q := `time() > 1400 default (
label_set(123, "foo", "bar"),
label_set(456, "__name__", "xxx"),
)`
r := netstorage.Result{
MetricName: metricNameExpected,
Values: []float64{456, 456, 456, 1600, 1800, 2000},
Timestamps: timestampsExpected,
}
resultExpected := []netstorage.Result{r}
f(q, resultExpected)
})
t.Run(`scalar default NaN`, func(t *testing.T) {
t.Parallel()
q := `time() > 1400 default (time() < -100)`
@ -5923,6 +5954,80 @@ func TestExecSuccess(t *testing.T) {
resultExpected := []netstorage.Result{r1}
f(q, resultExpected)
})
t.Run(`vector2 if vector1`, func(t *testing.T) {
t.Parallel()
q := `(
label_set(time()/10, "x", "y"),
label_set(time(), "foo", "bar", "__name__", "x"),
) if (
label_set(time()>1400, "foo", "bar"),
)`
r := netstorage.Result{
MetricName: metricNameExpected,
Values: []float64{nan, nan, nan, 1600, 1800, 2000},
Timestamps: timestampsExpected,
}
r.MetricName.MetricGroup = []byte("x")
r.MetricName.Tags = []storage.Tag{{
Key: []byte("foo"),
Value: []byte("bar"),
}}
resultExpected := []netstorage.Result{r}
f(q, resultExpected)
})
t.Run(`vector2 if vector2`, func(t *testing.T) {
t.Parallel()
q := `sort((
label_set(time()/10, "x", "y"),
label_set(time(), "foo", "bar", "__name__", "x"),
) if (
label_set(time()>1400, "foo", "bar"),
label_set(time()<1400, "x", "y"),
))`
r1 := netstorage.Result{
MetricName: metricNameExpected,
Values: []float64{100, 120, nan, nan, nan, nan},
Timestamps: timestampsExpected,
}
r1.MetricName.Tags = []storage.Tag{{
Key: []byte("x"),
Value: []byte("y"),
}}
r2 := netstorage.Result{
MetricName: metricNameExpected,
Values: []float64{nan, nan, nan, 1600, 1800, 2000},
Timestamps: timestampsExpected,
}
r2.MetricName.MetricGroup = []byte("x")
r2.MetricName.Tags = []storage.Tag{{
Key: []byte("foo"),
Value: []byte("bar"),
}}
resultExpected := []netstorage.Result{r1, r2}
f(q, resultExpected)
})
t.Run(`scalar if vector1`, func(t *testing.T) {
t.Parallel()
q := `time() if (
label_set(123, "foo", "bar"),
)`
resultExpected := []netstorage.Result{}
f(q, resultExpected)
})
t.Run(`scalar if vector2`, func(t *testing.T) {
t.Parallel()
q := `time() if (
label_set(123, "foo", "bar"),
alias(time() > 1400, "xxx"),
)`
r := netstorage.Result{
MetricName: metricNameExpected,
Values: []float64{nan, nan, nan, 1600, 1800, 2000},
Timestamps: timestampsExpected,
}
resultExpected := []netstorage.Result{r}
f(q, resultExpected)
})
t.Run(`if-default`, func(t *testing.T) {
t.Parallel()
q := `time() if time() > 1400 default -time()`

View file

@ -28,6 +28,7 @@ The following tip changes can be tested by building VictoriaMetrics components f
* BUGFIX: [vmctl](https://docs.victoriametrics.com/vmctl.html): return non-zero exit code on error. This allows handling `vmctl` errors in shell scripts. Previously `vmctl` was returning 0 exit code on error. See [this issue](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/2322).
* BUGFIX: [vmagent](https://docs.victoriametrics.com/vmagent.html): properly show `scrape_timeout` and `scrape_interval` options at `http://vmagent:8429/config` page. Previously these options weren't displayed even if they were set in `-promscrape.config`.
* BUGFIX: [MetricsQL](https://docs.victoriametrics.com/MetricsQL.html): properly handle joins on time series filtered by values. For example, `kube_pod_container_resource_requests{resource="cpu"} * on (namespace,pod) group_left() (kube_pod_status_phase{phase=~"Pending|Running"}==1)`. This query may result in `duplicate time series on the right side` error even if `==1` filter leaves only a single time series per `(namespace,pod)` labels. Now such query is properly executed.
* BUGFIX: [MetricsQL](https://docs.victoriametrics.com/MetricsQL.html): properly handle `scalar default vector`, `scalar if vector` and `scalar ifnot vector` queries. Previously such queries could return unexpected results from the `vector` part.
## [v1.76.1](https://github.com/VictoriaMetrics/VictoriaMetrics/releases/tag/v1.76.1)