mirror of
https://github.com/VictoriaMetrics/VictoriaMetrics.git
synced 2024-12-31 15:06:26 +00:00
wip
This commit is contained in:
parent
b7c062ac61
commit
a53b5570eb
5 changed files with 55 additions and 3 deletions
|
@ -19,6 +19,7 @@ according to [these docs](https://docs.victoriametrics.com/victorialogs/quicksta
|
||||||
|
|
||||||
## tip
|
## tip
|
||||||
|
|
||||||
|
* FEATURE: add `default` operator to [`math` pipe](https://docs.victoriametrics.com/victorialogs/logsql/#math-pipe). It allows setting `NaN` result to the given default value.
|
||||||
* FEATURE: allow omitting result name in [`stats` pipe](https://docs.victoriametrics.com/victorialogs/logsql/#stats-pipe). In this case the result name is automatically set to string representation of the corresponding [stats function expression](https://docs.victoriametrics.com/victorialogs/logsql/#stats-pipe-functions). For example, `_time:5m | count(*)` is valid [LogsQL query](https://docs.victoriametrics.com/victorialogs/logsql/) now. It is equivalent to `_time:5m | stats count(*) as "count(*)"`.
|
* FEATURE: allow omitting result name in [`stats` pipe](https://docs.victoriametrics.com/victorialogs/logsql/#stats-pipe). In this case the result name is automatically set to string representation of the corresponding [stats function expression](https://docs.victoriametrics.com/victorialogs/logsql/#stats-pipe-functions). For example, `_time:5m | count(*)` is valid [LogsQL query](https://docs.victoriametrics.com/victorialogs/logsql/) now. It is equivalent to `_time:5m | stats count(*) as "count(*)"`.
|
||||||
|
|
||||||
* BUGFIX: properly calculate the number of matching rows in `* | field_values x | stats count() rows` and in `* | unroll (x) | stats count() rows` queries.
|
* BUGFIX: properly calculate the number of matching rows in `* | field_values x | stats count() rows` and in `* | unroll (x) | stats count() rows` queries.
|
||||||
|
|
|
@ -1608,6 +1608,7 @@ The following mathematical operations are supported by `math` pipe:
|
||||||
- `arg1 / arg2` - divides `arg1` by `arg2`
|
- `arg1 / arg2` - divides `arg1` by `arg2`
|
||||||
- `arg1 % arg2` - returns the remainder of the division of `arg1` by `arg2`
|
- `arg1 % arg2` - returns the remainder of the division of `arg1` by `arg2`
|
||||||
- `arg1 ^ arg2` - returns the power of `arg1` by `arg2`
|
- `arg1 ^ arg2` - returns the power of `arg1` by `arg2`
|
||||||
|
- `arg1 default arg2` - returns `arg2` if `arg1` equals to `NaN`.
|
||||||
- `abs(arg)` - returns an absolute value for the given `arg`
|
- `abs(arg)` - returns an absolute value for the given `arg`
|
||||||
- `max(arg1, ..., argN)` - returns the maximum value among the given `arg1`, ..., `argN`
|
- `max(arg1, ..., argN)` - returns the maximum value among the given `arg1`, ..., `argN`
|
||||||
- `min(arg1, ..., argN)` - returns the minimum value among the given `arg1`, ..., `argN`
|
- `min(arg1, ..., argN)` - returns the minimum value among the given `arg1`, ..., `argN`
|
||||||
|
|
|
@ -161,6 +161,10 @@ var mathBinaryOps = map[string]mathBinaryOp{
|
||||||
priority: 3,
|
priority: 3,
|
||||||
f: mathFuncMinus,
|
f: mathFuncMinus,
|
||||||
},
|
},
|
||||||
|
"default": {
|
||||||
|
priority: 10,
|
||||||
|
f: mathFuncDefault,
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
type mathBinaryOp struct {
|
type mathBinaryOp struct {
|
||||||
|
@ -700,6 +704,18 @@ func mathFuncPow(result []float64, args [][]float64) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func mathFuncDefault(result []float64, args [][]float64) {
|
||||||
|
values := args[0]
|
||||||
|
defaultValues := args[1]
|
||||||
|
for i := range result {
|
||||||
|
f := values[i]
|
||||||
|
if math.IsNaN(f) {
|
||||||
|
f = defaultValues[i]
|
||||||
|
}
|
||||||
|
result[i] = f
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func mathFuncAbs(result []float64, args [][]float64) {
|
func mathFuncAbs(result []float64, args [][]float64) {
|
||||||
arg := args[0]
|
arg := args[0]
|
||||||
for i := range result {
|
for i := range result {
|
||||||
|
|
|
@ -22,6 +22,7 @@ func TestParsePipeMathSuccess(t *testing.T) {
|
||||||
f(`math min(3, foo, (1 + bar) / baz) as a, max(a, b) as b, (abs(c) + 5) as d`)
|
f(`math min(3, foo, (1 + bar) / baz) as a, max(a, b) as b, (abs(c) + 5) as d`)
|
||||||
f(`math round(foo) as x`)
|
f(`math round(foo) as x`)
|
||||||
f(`math round(foo, 0.1) as y`)
|
f(`math round(foo, 0.1) as y`)
|
||||||
|
f(`math (a / b default 10) as z`)
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestParsePipeMathFailure(t *testing.T) {
|
func TestParsePipeMathFailure(t *testing.T) {
|
||||||
|
@ -63,6 +64,39 @@ func TestPipeMath(t *testing.T) {
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
|
||||||
|
f("math a / b default 10 as c", [][]Field{
|
||||||
|
{
|
||||||
|
{"a", "v1"},
|
||||||
|
{"b", "2"},
|
||||||
|
{"c", "3"},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
{"a", "0"},
|
||||||
|
{"b", "0"},
|
||||||
|
{"c", "3"},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
{"a", "3"},
|
||||||
|
{"b", "2"},
|
||||||
|
},
|
||||||
|
}, [][]Field{
|
||||||
|
{
|
||||||
|
{"a", "v1"},
|
||||||
|
{"b", "2"},
|
||||||
|
{"c", "10"},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
{"a", "0"},
|
||||||
|
{"b", "0"},
|
||||||
|
{"c", "10"},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
{"a", "3"},
|
||||||
|
{"b", "2"},
|
||||||
|
{"c", "1.5"},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
f("math 1 as a", [][]Field{
|
f("math 1 as a", [][]Field{
|
||||||
{
|
{
|
||||||
{"a", "v1"},
|
{"a", "v1"},
|
||||||
|
|
|
@ -39,8 +39,8 @@ func TestPipeStats(t *testing.T) {
|
||||||
expectPipeResults(t, pipeStr, rows, rowsExpected)
|
expectPipeResults(t, pipeStr, rows, rowsExpected)
|
||||||
}
|
}
|
||||||
|
|
||||||
// missing 'stats' keyword
|
// missing 'stats' keyword and resutl name
|
||||||
f("count(*) as rows", [][]Field{
|
f("count(*)", [][]Field{
|
||||||
{
|
{
|
||||||
{"_msg", `abc`},
|
{"_msg", `abc`},
|
||||||
{"a", `2`},
|
{"a", `2`},
|
||||||
|
@ -56,7 +56,7 @@ func TestPipeStats(t *testing.T) {
|
||||||
},
|
},
|
||||||
}, [][]Field{
|
}, [][]Field{
|
||||||
{
|
{
|
||||||
{"rows", "3"},
|
{`count(*)`, "3"},
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue