This commit is contained in:
Aliaksandr Valialkin 2024-05-03 14:03:17 +02:00
parent 85a60e0743
commit 4ce2e32d4c
No known key found for this signature in database
GPG key ID: 52C003EE2BCDB9EB
9 changed files with 115 additions and 127 deletions

View file

@ -834,47 +834,63 @@ func TestParseQuerySuccess(t *testing.T) {
f(`foo | skip 10 | skip 100`, `foo | skip 10 | skip 100`) f(`foo | skip 10 | skip 100`, `foo | skip 10 | skip 100`)
// stats pipe count // stats pipe count
f(`* | Stats count() AS foo`, `* | stats count() as foo`)
f(`* | STATS bY (foo, b.a/r, "b az") count(*) XYz`, `* | stats by (foo, "b.a/r", "b az") count(*) as XYz`) f(`* | STATS bY (foo, b.a/r, "b az") count(*) XYz`, `* | stats by (foo, "b.a/r", "b az") count(*) as XYz`)
f(`* | stats by() COUNT(x, 'a).b,c|d') as qwert`, `* | stats count(x, "a).b,c|d") as qwert`) f(`* | stats by() COUNT(x, 'a).b,c|d') as qwert`, `* | stats count(x, "a).b,c|d") as qwert`)
f(`* | stats count() x`, `* | stats count(*) as x`)
f(`* | stats count(*) x`, `* | stats count(*) as x`)
f(`* | stats count(foo,*,bar) x`, `* | stats count(*) as x`)
// stats pipe sum // stats pipe sum
f(`* | stats Sum(foo) bar`, `* | stats sum(foo) as bar`) f(`* | stats Sum(foo) bar`, `* | stats sum(foo) as bar`)
f(`* | stats BY(x, y, ) SUM(foo,bar,) bar`, `* | stats by (x, y) sum(foo, bar) as bar`) f(`* | stats BY(x, y, ) SUM(foo,bar,) bar`, `* | stats by (x, y) sum(foo, bar) as bar`)
f(`* | stats sum() x`, `* | stats sum(*) as x`)
f(`* | stats sum(*) x`, `* | stats sum(*) as x`)
f(`* | stats sum(foo,*,bar) x`, `* | stats sum(*) as x`)
// stats pipe max // stats pipe max
f(`* | stats Max(foo) bar`, `* | stats max(foo) as bar`) f(`* | stats Max(foo) bar`, `* | stats max(foo) as bar`)
f(`* | stats BY(x, y, ) MAX(foo,bar,) bar`, `* | stats by (x, y) max(foo, bar) as bar`) f(`* | stats BY(x, y, ) MAX(foo,bar,) bar`, `* | stats by (x, y) max(foo, bar) as bar`)
f(`* | stats max() x`, `* | stats max(*) as x`)
f(`* | stats max(*) x`, `* | stats max(*) as x`)
f(`* | stats max(foo,*,bar) x`, `* | stats max(*) as x`)
// stats pipe min // stats pipe min
f(`* | stats Min(foo) bar`, `* | stats min(foo) as bar`) f(`* | stats Min(foo) bar`, `* | stats min(foo) as bar`)
f(`* | stats BY(x, y, ) MIN(foo,bar,) bar`, `* | stats by (x, y) min(foo, bar) as bar`) f(`* | stats BY(x, y, ) MIN(foo,bar,) bar`, `* | stats by (x, y) min(foo, bar) as bar`)
f(`* | stats min() x`, `* | stats min(*) as x`)
f(`* | stats min(*) x`, `* | stats min(*) as x`)
f(`* | stats min(foo,*,bar) x`, `* | stats min(*) as x`)
// stats pipe avg // stats pipe avg
f(`* | stats Avg(foo) bar`, `* | stats avg(foo) as bar`) f(`* | stats Avg(foo) bar`, `* | stats avg(foo) as bar`)
f(`* | stats BY(x, y, ) AVG(foo,bar,) bar`, `* | stats by (x, y) avg(foo, bar) as bar`) f(`* | stats BY(x, y, ) AVG(foo,bar,) bar`, `* | stats by (x, y) avg(foo, bar) as bar`)
f(`* | stats avg() x`, `* | stats avg(*) as x`)
f(`* | stats avg(*) x`, `* | stats avg(*) as x`)
f(`* | stats avg(foo,*,bar) x`, `* | stats avg(*) as x`)
// stats pipe uniq // stats pipe uniq
f(`* | stats uniq(foo) bar`, `* | stats uniq(foo) as bar`) f(`* | stats uniq(foo) bar`, `* | stats uniq(foo) as bar`)
f(`* | stats by(x, y) uniq(foo,bar) as baz`, `* | stats by (x, y) uniq(foo, bar) as baz`) f(`* | stats by(x, y) uniq(foo,bar) as baz`, `* | stats by (x, y) uniq(foo, bar) as baz`)
f(`* | stats by(x) uniq(*) z`, `* | stats by (x) uniq(*) as z`) f(`* | stats by(x) uniq(*) z`, `* | stats by (x) uniq(*) as z`)
f(`* | stats by(x) uniq() z`, `* | stats by (x) uniq() as z`) f(`* | stats by(x) uniq() z`, `* | stats by (x) uniq(*) as z`)
f(`* | stats by(x) uniq(a,*,b) z`, `* | stats by (x) uniq(*) as z`)
// stats pipe uniq_array // stats pipe uniq_array
f(`* | stats uniq_array(foo) bar`, `* | stats uniq_array(foo) as bar`) f(`* | stats uniq_array(foo) bar`, `* | stats uniq_array(foo) as bar`)
f(`* | stats by(x, y) uniq_array(foo, bar) as baz`, `* | stats by (x, y) uniq_array(foo, bar) as baz`) f(`* | stats by(x, y) uniq_array(foo, bar) as baz`, `* | stats by (x, y) uniq_array(foo, bar) as baz`)
f(`* | stats by(x) uniq_array(*) y`, `* | stats by (x) uniq_array(*) as y`) f(`* | stats by(x) uniq_array(*) y`, `* | stats by (x) uniq_array(*) as y`)
f(`* | stats by(x) uniq_array() y`, `* | stats by (x) uniq_array() as y`) f(`* | stats by(x) uniq_array() y`, `* | stats by (x) uniq_array(*) as y`)
f(`* | stats by(x) uniq_array(a,*,b) y`, `* | stats by (x) uniq_array(*) as y`)
// stats pipe multiple funcs // stats pipe multiple funcs
f(`* | stats count() "foo.bar:baz", uniq(a) bar`, `* | stats count() as "foo.bar:baz", uniq(a) as bar`) f(`* | stats count() "foo.bar:baz", uniq(a) bar`, `* | stats count(*) as "foo.bar:baz", uniq(a) as bar`)
f(`* | stats by (x, y) count(*) foo, uniq(a,b) bar`, `* | stats by (x, y) count(*) as foo, uniq(a, b) as bar`) f(`* | stats by (x, y) count(*) foo, uniq(a,b) bar`, `* | stats by (x, y) count(*) as foo, uniq(a, b) as bar`)
// stats pipe with grouping buckets // stats pipe with grouping buckets
f(`* | stats by(_time:1d, response_size:1_000KiB, request_duration:5s, foo) count() as foo`, `* | stats by (_time:1d, response_size:1_000KiB, request_duration:5s, foo) count() as foo`) f(`* | stats by(_time:1d, response_size:1_000KiB, request_duration:5s, foo) count() as foo`, `* | stats by (_time:1d, response_size:1_000KiB, request_duration:5s, foo) count(*) as foo`)
f(`*|stats by(client_ip:/24, server_ip:/16) count() foo`, `* | stats by (client_ip:/24, server_ip:/16) count() as foo`) f(`*|stats by(client_ip:/24, server_ip:/16) count() foo`, `* | stats by (client_ip:/24, server_ip:/16) count(*) as foo`)
f(`* | stats by(_time:1d offset 2h) count() as foo`, `* | stats by (_time:1d offset 2h) count() as foo`) f(`* | stats by(_time:1d offset 2h) count() as foo`, `* | stats by (_time:1d offset 2h) count(*) as foo`)
f(`* | stats by(_time:1d offset -2.5h5m) count() as foo`, `* | stats by (_time:1d offset -2.5h5m) count() as foo`) f(`* | stats by(_time:1d offset -2.5h5m) count() as foo`, `* | stats by (_time:1d offset -2.5h5m) count(*) as foo`)
// multiple different pipes // multiple different pipes
f(`* | fields foo, bar | head 100 | stats by(foo,bar) count(baz) as qwert`, `* | fields foo, bar | head 100 | stats by (foo, bar) count(baz) as qwert`) f(`* | fields foo, bar | head 100 | stats by(foo,bar) count(baz) as qwert`, `* | fields foo, bar | head 100 | stats by (foo, bar) count(baz) as qwert`)
@ -1123,22 +1139,18 @@ func TestParseQueryFailure(t *testing.T) {
// invalid stats sum // invalid stats sum
f(`foo | stats sum`) f(`foo | stats sum`)
f(`foo | stats sum()`) f(`foo | stats sum()`)
f(`foo | stats sum() as abc`)
// invalid stats max // invalid stats max
f(`foo | stats max`) f(`foo | stats max`)
f(`foo | stats max()`) f(`foo | stats max()`)
f(`foo | stats max() as abc`)
// invalid stats min // invalid stats min
f(`foo | stats min`) f(`foo | stats min`)
f(`foo | stats min()`) f(`foo | stats min()`)
f(`foo | stats min() as abc`)
// invalid stats avg // invalid stats avg
f(`foo | stats avg`) f(`foo | stats avg`)
f(`foo | stats avg()`) f(`foo | stats avg()`)
f(`foo | stats avg() as abc`)
// invalid stats uniq // invalid stats uniq
f(`foo | stats uniq`) f(`foo | stats uniq`)

View file

@ -2,6 +2,7 @@ package logstorage
import ( import (
"fmt" "fmt"
"slices"
"strings" "strings"
"sync/atomic" "sync/atomic"
"unsafe" "unsafe"
@ -659,7 +660,7 @@ func tryParseBucketSize(s string) (float64, bool) {
return 0, false return 0, false
} }
func parseFieldNamesForFunc(lex *lexer, funcName string) ([]string, error) { func parseFieldNamesForStatsFunc(lex *lexer, funcName string) ([]string, error) {
if !lex.isKeyword(funcName) { if !lex.isKeyword(funcName) {
return nil, fmt.Errorf("unexpected func; got %q; want %q", lex.token, funcName) return nil, fmt.Errorf("unexpected func; got %q; want %q", lex.token, funcName)
} }
@ -668,6 +669,9 @@ func parseFieldNamesForFunc(lex *lexer, funcName string) ([]string, error) {
if err != nil { if err != nil {
return nil, fmt.Errorf("cannot parse %q args: %w", funcName, err) return nil, fmt.Errorf("cannot parse %q args: %w", funcName, err)
} }
if len(fields) == 0 || slices.Contains(fields, "*") {
fields = []string{"*"}
}
return fields, nil return fields, nil
} }

View file

@ -1,7 +1,6 @@
package logstorage package logstorage
import ( import (
"fmt"
"math" "math"
"slices" "slices"
"strconv" "strconv"
@ -43,9 +42,7 @@ func (sap *statsAvgProcessor) updateStatsForAllRows(br *blockResult) int {
sap.sum += f sap.sum += f
sap.count += uint64(count) sap.count += uint64(count)
} }
return 0 } else {
}
// Scan the requested columns // Scan the requested columns
for _, field := range sap.sa.fields { for _, field := range sap.sa.fields {
c := br.getColumnByName(field) c := br.getColumnByName(field)
@ -53,6 +50,7 @@ func (sap *statsAvgProcessor) updateStatsForAllRows(br *blockResult) int {
sap.sum += f sap.sum += f
sap.count += uint64(count) sap.count += uint64(count)
} }
}
return 0 return 0
} }
@ -66,9 +64,7 @@ func (sap *statsAvgProcessor) updateStatsForRow(br *blockResult, rowIdx int) int
sap.count++ sap.count++
} }
} }
return 0 } else {
}
// Scan only the given fields for the given row // Scan only the given fields for the given row
for _, field := range sap.sa.fields { for _, field := range sap.sa.fields {
c := br.getColumnByName(field) c := br.getColumnByName(field)
@ -78,6 +74,7 @@ func (sap *statsAvgProcessor) updateStatsForRow(br *blockResult, rowIdx int) int
sap.count++ sap.count++
} }
} }
}
return 0 return 0
} }
@ -93,13 +90,10 @@ func (sap *statsAvgProcessor) finalizeStats() string {
} }
func parseStatsAvg(lex *lexer) (*statsAvg, error) { func parseStatsAvg(lex *lexer) (*statsAvg, error) {
fields, err := parseFieldNamesForFunc(lex, "avg") fields, err := parseFieldNamesForStatsFunc(lex, "avg")
if err != nil { if err != nil {
return nil, err return nil, err
} }
if len(fields) == 0 {
return nil, fmt.Errorf("'avg' must contain at least one arg")
}
sa := &statsAvg{ sa := &statsAvg{
fields: fields, fields: fields,
containsStar: slices.Contains(fields, "*"), containsStar: slices.Contains(fields, "*"),

View file

@ -36,7 +36,7 @@ type statsCountProcessor struct {
func (scp *statsCountProcessor) updateStatsForAllRows(br *blockResult) int { func (scp *statsCountProcessor) updateStatsForAllRows(br *blockResult) int {
fields := scp.sc.fields fields := scp.sc.fields
if len(fields) == 0 || scp.sc.containsStar { if scp.sc.containsStar {
// Fast path - unconditionally count all the columns. // Fast path - unconditionally count all the columns.
scp.rowsCount += uint64(len(br.timestamps)) scp.rowsCount += uint64(len(br.timestamps))
return 0 return 0
@ -134,7 +134,7 @@ func (scp *statsCountProcessor) updateStatsForAllRows(br *blockResult) int {
func (scp *statsCountProcessor) updateStatsForRow(br *blockResult, rowIdx int) int { func (scp *statsCountProcessor) updateStatsForRow(br *blockResult, rowIdx int) int {
fields := scp.sc.fields fields := scp.sc.fields
if len(fields) == 0 || scp.sc.containsStar { if scp.sc.containsStar {
// Fast path - unconditionally count the given column // Fast path - unconditionally count the given column
scp.rowsCount++ scp.rowsCount++
return 0 return 0
@ -194,7 +194,7 @@ func (scp *statsCountProcessor) finalizeStats() string {
} }
func parseStatsCount(lex *lexer) (*statsCount, error) { func parseStatsCount(lex *lexer) (*statsCount, error) {
fields, err := parseFieldNamesForFunc(lex, "count") fields, err := parseFieldNamesForStatsFunc(lex, "count")
if err != nil { if err != nil {
return nil, err return nil, err
} }

View file

@ -1,7 +1,6 @@
package logstorage package logstorage
import ( import (
"fmt"
"math" "math"
"slices" "slices"
"strconv" "strconv"
@ -44,9 +43,7 @@ func (smp *statsMaxProcessor) updateStatsForAllRows(br *blockResult) int {
smp.max = f smp.max = f
} }
} }
return 0 } else {
}
// Find the maximum value across the requested columns // Find the maximum value across the requested columns
for _, field := range smp.sm.fields { for _, field := range smp.sm.fields {
c := br.getColumnByName(field) c := br.getColumnByName(field)
@ -55,6 +52,7 @@ func (smp *statsMaxProcessor) updateStatsForAllRows(br *blockResult) int {
smp.max = f smp.max = f
} }
} }
}
return 0 return 0
} }
@ -67,9 +65,7 @@ func (smp *statsMaxProcessor) updateStatsForRow(br *blockResult, rowIdx int) int
smp.max = f smp.max = f
} }
} }
return 0 } else {
}
// Find the maximum value across the requested fields for the given row // Find the maximum value across the requested fields for the given row
for _, field := range smp.sm.fields { for _, field := range smp.sm.fields {
c := br.getColumnByName(field) c := br.getColumnByName(field)
@ -78,6 +74,7 @@ func (smp *statsMaxProcessor) updateStatsForRow(br *blockResult, rowIdx int) int
smp.max = f smp.max = f
} }
} }
}
return 0 return 0
} }
@ -93,13 +90,10 @@ func (smp *statsMaxProcessor) finalizeStats() string {
} }
func parseStatsMax(lex *lexer) (*statsMax, error) { func parseStatsMax(lex *lexer) (*statsMax, error) {
fields, err := parseFieldNamesForFunc(lex, "max") fields, err := parseFieldNamesForStatsFunc(lex, "max")
if err != nil { if err != nil {
return nil, err return nil, err
} }
if len(fields) == 0 {
return nil, fmt.Errorf("'max' must contain at least one arg")
}
sm := &statsMax{ sm := &statsMax{
fields: fields, fields: fields,
containsStar: slices.Contains(fields, "*"), containsStar: slices.Contains(fields, "*"),

View file

@ -1,7 +1,6 @@
package logstorage package logstorage
import ( import (
"fmt"
"math" "math"
"slices" "slices"
"strconv" "strconv"
@ -44,9 +43,7 @@ func (smp *statsMinProcessor) updateStatsForAllRows(br *blockResult) int {
smp.min = f smp.min = f
} }
} }
return 0 } else {
}
// Find the minimum value across the requested columns // Find the minimum value across the requested columns
for _, field := range smp.sm.fields { for _, field := range smp.sm.fields {
c := br.getColumnByName(field) c := br.getColumnByName(field)
@ -55,6 +52,7 @@ func (smp *statsMinProcessor) updateStatsForAllRows(br *blockResult) int {
smp.min = f smp.min = f
} }
} }
}
return 0 return 0
} }
@ -67,9 +65,7 @@ func (smp *statsMinProcessor) updateStatsForRow(br *blockResult, rowIdx int) int
smp.min = f smp.min = f
} }
} }
return 0 } else {
}
// Find the minimum value across the requested fields for the given row // Find the minimum value across the requested fields for the given row
for _, field := range smp.sm.fields { for _, field := range smp.sm.fields {
c := br.getColumnByName(field) c := br.getColumnByName(field)
@ -78,6 +74,7 @@ func (smp *statsMinProcessor) updateStatsForRow(br *blockResult, rowIdx int) int
smp.min = f smp.min = f
} }
} }
}
return 0 return 0
} }
@ -93,13 +90,10 @@ func (smp *statsMinProcessor) finalizeStats() string {
} }
func parseStatsMin(lex *lexer) (*statsMin, error) { func parseStatsMin(lex *lexer) (*statsMin, error) {
fields, err := parseFieldNamesForFunc(lex, "min") fields, err := parseFieldNamesForStatsFunc(lex, "min")
if err != nil { if err != nil {
return nil, err return nil, err
} }
if len(fields) == 0 {
return nil, fmt.Errorf("'min' must contain at least one arg")
}
sm := &statsMin{ sm := &statsMin{
fields: fields, fields: fields,
containsStar: slices.Contains(fields, "*"), containsStar: slices.Contains(fields, "*"),

View file

@ -1,7 +1,6 @@
package logstorage package logstorage
import ( import (
"fmt"
"math" "math"
"slices" "slices"
"strconv" "strconv"
@ -48,9 +47,7 @@ func (ssp *statsSumProcessor) updateStatsForAllRows(br *blockResult) int {
} }
} }
} }
return 0 } else {
}
// Sum the requested columns // Sum the requested columns
for _, field := range ssp.ss.fields { for _, field := range ssp.ss.fields {
c := br.getColumnByName(field) c := br.getColumnByName(field)
@ -63,6 +60,7 @@ func (ssp *statsSumProcessor) updateStatsForAllRows(br *blockResult) int {
} }
} }
} }
}
return 0 return 0
} }
@ -79,9 +77,7 @@ func (ssp *statsSumProcessor) updateStatsForRow(br *blockResult, rowIdx int) int
} }
} }
} }
return 0 } else {
}
// Sum only the given fields for the given row // Sum only the given fields for the given row
for _, field := range ssp.ss.fields { for _, field := range ssp.ss.fields {
c := br.getColumnByName(field) c := br.getColumnByName(field)
@ -94,6 +90,7 @@ func (ssp *statsSumProcessor) updateStatsForRow(br *blockResult, rowIdx int) int
} }
} }
} }
}
return 0 return 0
} }
@ -107,13 +104,10 @@ func (ssp *statsSumProcessor) finalizeStats() string {
} }
func parseStatsSum(lex *lexer) (*statsSum, error) { func parseStatsSum(lex *lexer) (*statsSum, error) {
fields, err := parseFieldNamesForFunc(lex, "sum") fields, err := parseFieldNamesForStatsFunc(lex, "sum")
if err != nil { if err != nil {
return nil, err return nil, err
} }
if len(fields) == 0 {
return nil, fmt.Errorf("'sum' must contain at least one arg")
}
ss := &statsSum{ ss := &statsSum{
fields: fields, fields: fields,
containsStar: slices.Contains(fields, "*"), containsStar: slices.Contains(fields, "*"),

View file

@ -45,7 +45,7 @@ func (sup *statsUniqProcessor) updateStatsForAllRows(br *blockResult) int {
m := sup.m m := sup.m
stateSizeIncrease := 0 stateSizeIncrease := 0
if len(fields) == 0 || sup.su.containsStar { if sup.su.containsStar {
// Count unique rows // Count unique rows
columns := br.getColumns() columns := br.getColumns()
keyBuf := sup.keyBuf[:0] keyBuf := sup.keyBuf[:0]
@ -222,7 +222,7 @@ func (sup *statsUniqProcessor) updateStatsForRow(br *blockResult, rowIdx int) in
m := sup.m m := sup.m
stateSizeIncrease := 0 stateSizeIncrease := 0
if len(fields) == 0 || sup.su.containsStar { if sup.su.containsStar {
// Count unique rows // Count unique rows
allEmptyValues := true allEmptyValues := true
keyBuf := sup.keyBuf[:0] keyBuf := sup.keyBuf[:0]
@ -358,7 +358,7 @@ func (sup *statsUniqProcessor) finalizeStats() string {
} }
func parseStatsUniq(lex *lexer) (*statsUniq, error) { func parseStatsUniq(lex *lexer) (*statsUniq, error) {
fields, err := parseFieldNamesForFunc(lex, "uniq") fields, err := parseFieldNamesForStatsFunc(lex, "uniq")
if err != nil { if err != nil {
return nil, err return nil, err
} }

View file

@ -39,16 +39,14 @@ type statsUniqArrayProcessor struct {
} }
func (sup *statsUniqArrayProcessor) updateStatsForAllRows(br *blockResult) int { func (sup *statsUniqArrayProcessor) updateStatsForAllRows(br *blockResult) int {
fields := sup.su.fields
stateSizeIncrease := 0 stateSizeIncrease := 0
if len(fields) == 0 || sup.su.containsStar { if sup.su.containsStar {
columns := br.getColumns() columns := br.getColumns()
for i := range columns { for i := range columns {
stateSizeIncrease += sup.updateStatsForAllRowsColumn(&columns[i], br) stateSizeIncrease += sup.updateStatsForAllRowsColumn(&columns[i], br)
} }
} else { } else {
for _, field := range fields { for _, field := range sup.su.fields {
c := br.getColumnByName(field) c := br.getColumnByName(field)
stateSizeIncrease += sup.updateStatsForAllRowsColumn(&c, br) stateSizeIncrease += sup.updateStatsForAllRowsColumn(&c, br)
} }
@ -110,16 +108,14 @@ func (sup *statsUniqArrayProcessor) updateStatsForAllRowsColumn(c *blockResultCo
} }
func (sup *statsUniqArrayProcessor) updateStatsForRow(br *blockResult, rowIdx int) int { func (sup *statsUniqArrayProcessor) updateStatsForRow(br *blockResult, rowIdx int) int {
fields := sup.su.fields
stateSizeIncrease := 0 stateSizeIncrease := 0
if len(fields) == 0 || sup.su.containsStar { if sup.su.containsStar {
columns := br.getColumns() columns := br.getColumns()
for i := range columns { for i := range columns {
stateSizeIncrease += sup.updateStatsForRowColumn(&columns[i], br, rowIdx) stateSizeIncrease += sup.updateStatsForRowColumn(&columns[i], br, rowIdx)
} }
} else { } else {
for _, field := range fields { for _, field := range sup.su.fields {
c := br.getColumnByName(field) c := br.getColumnByName(field)
stateSizeIncrease += sup.updateStatsForRowColumn(&c, br, rowIdx) stateSizeIncrease += sup.updateStatsForRowColumn(&c, br, rowIdx)
} }
@ -219,7 +215,7 @@ func (sup *statsUniqArrayProcessor) finalizeStats() string {
} }
func parseStatsUniqArray(lex *lexer) (*statsUniqArray, error) { func parseStatsUniqArray(lex *lexer) (*statsUniqArray, error) {
fields, err := parseFieldNamesForFunc(lex, "uniq_array") fields, err := parseFieldNamesForStatsFunc(lex, "uniq_array")
if err != nil { if err != nil {
return nil, err return nil, err
} }