This commit is contained in:
Aliaksandr Valialkin 2024-05-15 22:31:21 +02:00
parent 3010034c7a
commit d50273d6c0
No known key found for this signature in database
GPG key ID: 52C003EE2BCDB9EB
5 changed files with 33 additions and 6 deletions

View file

@ -1200,6 +1200,12 @@ The reverse order can be applied globally via `desc` keyword after `by(...)` cla
_time:5m | sort by (foo, bar) desc _time:5m | sort by (foo, bar) desc
``` ```
The `by` keyword can be skipped in `sort ...` pipe. For example, the following query is equivalent to the previous one:
```logsql
_time:5m | sort (foo, bar) desc
```
Sorting of big number of logs can consume a lot of CPU time and memory. Sometimes it is enough to return the first `N` entries with the biggest Sorting of big number of logs can consume a lot of CPU time and memory. Sometimes it is enough to return the first `N` entries with the biggest
or the smallest values. This can be done by adding `limit N` to the end of `sort ...` pipe. or the smallest values. This can be done by adding `limit N` to the end of `sort ...` pipe.
Such a query consumes lower amounts of memory when sorting big number of logs, since it keeps in memory only `N` log entries. Such a query consumes lower amounts of memory when sorting big number of logs, since it keeps in memory only `N` log entries.
@ -1257,6 +1263,12 @@ This allows limiting memory usage. For example, the following query returns up t
_time:5m | uniq by (host, path) limit 100 _time:5m | uniq by (host, path) limit 100
``` ```
The `by` keyword can be skipped in `uniq ...` pipe. For example, the following query is equivalent to the previous one:
```logsql
_time:5m | uniq (host, path) limit 100
```
See also: See also:
- [`uniq_values` stats function](#uniq_values-stats) - [`uniq_values` stats function](#uniq_values-stats)
@ -1316,6 +1328,12 @@ grouped by `(host, path)` fields:
_time:5m | stats by (host, path) count() logs_total, count_uniq(ip) ips_total _time:5m | stats by (host, path) count() logs_total, count_uniq(ip) ips_total
``` ```
The `by` keyword can be skipped in `stats ...` pipe. For example, the following query is equvalent to the previous one:
```logsql
_time:5m | stats (host, path) count() logs_total, count_uniq(ip) ips_total
```
#### Stats by time buckets #### Stats by time buckets
The following syntax can be used for calculating stats grouped by time buckets: The following syntax can be used for calculating stats grouped by time buckets:

View file

@ -966,6 +966,7 @@ func TestParseQuerySuccess(t *testing.T) {
f(`* | stats by (_time:week) count() foo`, `* | stats by (_time:week) count(*) as foo`) f(`* | stats by (_time:week) count() foo`, `* | stats by (_time:week) count(*) as foo`)
f(`* | stats by (_time:month) count() foo`, `* | stats by (_time:month) count(*) as foo`) f(`* | stats by (_time:month) count() foo`, `* | stats by (_time:month) count(*) as foo`)
f(`* | stats by (_time:year offset 6.5h) count() foo`, `* | stats by (_time:year offset 6.5h) count(*) as foo`) f(`* | stats by (_time:year offset 6.5h) count() foo`, `* | stats by (_time:year offset 6.5h) count(*) as foo`)
f(`* | stats (_time:year offset 6.5h) count() foo`, `* | stats by (_time:year offset 6.5h) count(*) as foo`)
// sort pipe // sort pipe
f(`* | sort`, `* | sort`) f(`* | sort`, `* | sort`)
@ -983,6 +984,7 @@ func TestParseQuerySuccess(t *testing.T) {
f(`* | sort by (foo desc, bar) desc limit 10`, `* | sort by (foo desc, bar) desc limit 10`) f(`* | sort by (foo desc, bar) desc limit 10`, `* | sort by (foo desc, bar) desc limit 10`)
f(`* | sort by (foo desc, bar) desc OFFSET 30 limit 10`, `* | sort by (foo desc, bar) desc offset 30 limit 10`) f(`* | sort by (foo desc, bar) desc OFFSET 30 limit 10`, `* | sort by (foo desc, bar) desc offset 30 limit 10`)
f(`* | sort by (foo desc, bar) desc limit 10 OFFSET 30`, `* | sort by (foo desc, bar) desc offset 30 limit 10`) f(`* | sort by (foo desc, bar) desc limit 10 OFFSET 30`, `* | sort by (foo desc, bar) desc offset 30 limit 10`)
f(`* | sort (foo desc, bar) desc limit 10 OFFSET 30`, `* | sort by (foo desc, bar) desc offset 30 limit 10`)
// uniq pipe // uniq pipe
f(`* | uniq`, `* | uniq`) f(`* | uniq`, `* | uniq`)
@ -991,6 +993,7 @@ func TestParseQuerySuccess(t *testing.T) {
f(`* | uniq by(foo,*,bar)`, `* | uniq`) f(`* | uniq by(foo,*,bar)`, `* | uniq`)
f(`* | uniq by(f1,f2)`, `* | uniq by (f1, f2)`) f(`* | uniq by(f1,f2)`, `* | uniq by (f1, f2)`)
f(`* | uniq by(f1,f2) limit 10`, `* | uniq by (f1, f2) limit 10`) f(`* | uniq by(f1,f2) limit 10`, `* | uniq by (f1, f2) limit 10`)
f(`* | uniq (f1,f2) limit 10`, `* | uniq by (f1, f2) limit 10`)
f(`* | uniq limit 10`, `* | uniq limit 10`) f(`* | uniq limit 10`, `* | uniq limit 10`)
// multiple different pipes // multiple different pipes

View file

@ -689,8 +689,10 @@ func parsePipeSort(lex *lexer) (*pipeSort, error) {
lex.nextToken() lex.nextToken()
var ps pipeSort var ps pipeSort
if lex.isKeyword("by", "(") {
if lex.isKeyword("by") { if lex.isKeyword("by") {
lex.nextToken() lex.nextToken()
}
bfs, err := parseBySortFields(lex) bfs, err := parseBySortFields(lex)
if err != nil { if err != nil {
return nil, fmt.Errorf("cannot parse 'by' clause: %w", err) return nil, fmt.Errorf("cannot parse 'by' clause: %w", err)

View file

@ -443,8 +443,10 @@ func parsePipeStats(lex *lexer) (*pipeStats, error) {
lex.nextToken() lex.nextToken()
var ps pipeStats var ps pipeStats
if lex.isKeyword("by", "(") {
if lex.isKeyword("by") { if lex.isKeyword("by") {
lex.nextToken() lex.nextToken()
}
bfs, err := parseByStatsFields(lex) bfs, err := parseByStatsFields(lex)
if err != nil { if err != nil {
return nil, fmt.Errorf("cannot parse 'by' clause: %w", err) return nil, fmt.Errorf("cannot parse 'by' clause: %w", err)

View file

@ -360,8 +360,10 @@ func parsePipeUniq(lex *lexer) (*pipeUniq, error) {
lex.nextToken() lex.nextToken()
var pu pipeUniq var pu pipeUniq
if lex.isKeyword("by", "(") {
if lex.isKeyword("by") { if lex.isKeyword("by") {
lex.nextToken() lex.nextToken()
}
bfs, err := parseFieldNamesInParens(lex) bfs, err := parseFieldNamesInParens(lex)
if err != nil { if err != nil {
return nil, fmt.Errorf("cannot parse 'by' clause: %w", err) return nil, fmt.Errorf("cannot parse 'by' clause: %w", err)