This commit is contained in:
Aliaksandr Valialkin 2024-05-20 16:41:08 +02:00
parent a8dde0aeac
commit ae0a11d7c1
No known key found for this signature in database
GPG key ID: 52C003EE2BCDB9EB
6 changed files with 88 additions and 13 deletions

View file

@ -320,10 +320,10 @@ func (q *Query) Optimize() {
switch t := p.(type) { switch t := p.(type) {
case *pipeStats: case *pipeStats:
for _, f := range t.funcs { for _, f := range t.funcs {
if f.iff != nil {
optimizeFilterIn(f.iff) optimizeFilterIn(f.iff)
} }
} case *pipeExtract:
optimizeFilterIn(t.iff)
} }
} }
} }
@ -345,6 +345,10 @@ func removeStarFilters(f filter) filter {
} }
func optimizeFilterIn(f filter) { func optimizeFilterIn(f filter) {
if f == nil {
return
}
visitFunc := func(f filter) bool { visitFunc := func(f filter) bool {
fi, ok := f.(*filterIn) fi, ok := f.(*filterIn)
if ok && fi.q != nil { if ok && fi.q != nil {

View file

@ -65,7 +65,10 @@ func parsePipes(lex *lexer) ([]pipe, error) {
var pipes []pipe var pipes []pipe
for !lex.isKeyword(")", "") { for !lex.isKeyword(")", "") {
if !lex.isKeyword("|") { if !lex.isKeyword("|") {
return nil, fmt.Errorf("expecting '|'; got %q", lex.token) if len(pipes) == 0 {
return nil, fmt.Errorf("expecting '|' after the query filters; got %q", lex.token)
}
return nil, fmt.Errorf("expecting '|' after [%s] pipe; got %q", pipes[len(pipes)-1], lex.token)
} }
lex.nextToken() lex.nextToken()
p, err := parsePipe(lex) p, err := parsePipe(lex)

View file

@ -205,7 +205,7 @@ func parsePipeExtract(lex *lexer) (*pipeExtract, error) {
if lex.isKeyword("if") { if lex.isKeyword("if") {
iff, err := parseIfFilter(lex) iff, err := parseIfFilter(lex)
if err != nil { if err != nil {
return nil, fmt.Errorf("cannot parse 'if' filter for %s: %w", pe, err) return nil, err
} }
pe.iff = iff pe.iff = iff

View file

@ -4,6 +4,35 @@ import (
"testing" "testing"
) )
func TestParsePipeExtractSuccess(t *testing.T) {
f := func(pipeStr string) {
t.Helper()
expectParsePipeSuccess(t, pipeStr)
}
f(`extract "foo<bar>"`)
f(`extract from x "foo<bar>"`)
f(`extract from x "foo<bar>" if (y:in(a:foo bar | uniq by (qwe) limit 10))`)
}
func TestParsePipeExtractFailure(t *testing.T) {
f := func(pipeStr string) {
t.Helper()
expectParsePipeFailure(t, pipeStr)
}
f(`extract`)
f(`extract from`)
f(`extract if (x:y)`)
f(`extract if (x:y) "a<b>"`)
f(`extract "a<b>" if`)
f(`extract "a<b>" if (foo`)
f(`extract "a<b>" if "foo"`)
f(`extract "a"`)
f(`extract "<a><b>"`)
f(`extract "<*>foo<_>bar"`)
}
func TestPipeExtract(t *testing.T) { func TestPipeExtract(t *testing.T) {
f := func(pipeStr string, rows, rowsExpected [][]Field) { f := func(pipeStr string, rows, rowsExpected [][]Field) {
t.Helper() t.Helper()
@ -211,3 +240,28 @@ func TestPipeExtractUpdateNeededFields(t *testing.T) {
f("extract from x '<foo>x<bar>'", "f2,foo,x,y", "", "f2,x,y", "") f("extract from x '<foo>x<bar>'", "f2,foo,x,y", "", "f2,x,y", "")
f("extract from x '<foo>x<bar>' if (a:b foo:q)", "f2,foo,x,y", "", "a,f2,foo,x,y", "") f("extract from x '<foo>x<bar>' if (a:b foo:q)", "f2,foo,x,y", "", "a,f2,foo,x,y", "")
} }
func expectParsePipeFailure(t *testing.T, pipeStr string) {
t.Helper()
lex := newLexer(pipeStr)
p, err := parsePipe(lex)
if err == nil {
t.Fatalf("expecting error when parsing [%s]; parsed result: [%s]", pipeStr, p)
}
}
func expectParsePipeSuccess(t *testing.T, pipeStr string) {
t.Helper()
lex := newLexer(pipeStr)
p, err := parsePipe(lex)
if err != nil {
t.Fatalf("cannot parse [%s]: %s", pipeStr, err)
}
pipeStrResult := p.String()
if pipeStrResult != pipeStr {
t.Fatalf("unexpected string representation of pipe; got\n%s\nwant\n%s", pipeStrResult, pipeStr)
}
}

View file

@ -526,7 +526,7 @@ func parsePipeStats(lex *lexer) (*pipeStats, error) {
if lex.isKeyword("if") { if lex.isKeyword("if") {
iff, err := parseIfFilter(lex) iff, err := parseIfFilter(lex)
if err != nil { if err != nil {
return nil, fmt.Errorf("cannot parse 'if' filter for %s: %w", sf, err) return nil, err
} }
f.iff = iff f.iff = iff
@ -537,7 +537,7 @@ func parsePipeStats(lex *lexer) (*pipeStats, error) {
resultName, err := parseResultName(lex) resultName, err := parseResultName(lex)
if err != nil { if err != nil {
return nil, fmt.Errorf("cannot parse result name for %s: %w", sf, err) return nil, fmt.Errorf("cannot parse result name for [%s]: %w", sf, err)
} }
f.resultName = resultName f.resultName = resultName

View file

@ -260,6 +260,9 @@ func (s *Storage) initFilterInValues(ctx context.Context, tenantIDs []TenantID,
} }
func hasFilterInWithQueryForFilter(f filter) bool { func hasFilterInWithQueryForFilter(f filter) bool {
if f == nil {
return false
}
visitFunc := func(f filter) bool { visitFunc := func(f filter) bool {
fi, ok := f.(*filterIn) fi, ok := f.(*filterIn)
return ok && fi.needExecuteQuery return ok && fi.needExecuteQuery
@ -269,12 +272,15 @@ func hasFilterInWithQueryForFilter(f filter) bool {
func hasFilterInWithQueryForPipes(pipes []pipe) bool { func hasFilterInWithQueryForPipes(pipes []pipe) bool {
for _, p := range pipes { for _, p := range pipes {
ps, ok := p.(*pipeStats) switch t := p.(type) {
if !ok { case *pipeStats:
continue for _, f := range t.funcs {
if hasFilterInWithQueryForFilter(f.iff) {
return true
} }
for _, f := range ps.funcs { }
if f.iff != nil && hasFilterInWithQueryForFilter(f.iff) { case *pipeExtract:
if hasFilterInWithQueryForFilter(t.iff) {
return true return true
} }
} }
@ -333,6 +339,14 @@ func initFilterInValuesForPipes(cache map[string][]string, pipes []pipe, getFiel
byFields: t.byFields, byFields: t.byFields,
funcs: funcsNew, funcs: funcsNew,
} }
case *pipeExtract:
fNew, err := initFilterInValuesForFilter(cache, t.iff, getFieldValuesFunc)
if err != nil {
return nil, err
}
pe := *t
pe.iff = fNew
pipesNew[i] = &pe
default: default:
pipesNew[i] = p pipesNew[i] = p
} }