VictoriaMetrics/lib/logstorage/pipe_extract_test.go

225 lines
5.4 KiB
Go
Raw Normal View History

2024-05-19 02:24:32 +00:00
package logstorage
import (
"reflect"
"testing"
)
func TestExtractFormatApply(t *testing.T) {
2024-05-19 13:11:17 +00:00
f := func(pattern, s string, resultsExpected []string) {
2024-05-19 02:24:32 +00:00
t.Helper()
2024-05-19 13:11:17 +00:00
steps, err := parseExtractFormatSteps(pattern)
2024-05-19 02:24:32 +00:00
if err != nil {
t.Fatalf("unexpected error: %s", err)
}
2024-05-19 10:56:08 +00:00
ef := newExtractFormat(steps)
2024-05-19 02:24:32 +00:00
ef.apply(s)
2024-05-19 10:56:08 +00:00
if len(ef.fields) != len(resultsExpected) {
t.Fatalf("unexpected number of results; got %d; want %d", len(ef.fields), len(resultsExpected))
}
for i, f := range ef.fields {
if v := *f.value; v != resultsExpected[i] {
t.Fatalf("unexpected value for field %q; got %q; want %q", f.name, v, resultsExpected[i])
}
2024-05-19 02:24:32 +00:00
}
}
f("<foo>", "", []string{""})
f("<foo>", "abc", []string{"abc"})
2024-05-19 10:56:08 +00:00
f("<foo>bar", "", []string{""})
f("<foo>bar", "bar", []string{""})
f("<foo>bar", "bazbar", []string{"baz"})
f("<foo>bar", "a bazbar xdsf", []string{"a baz"})
f("<foo>bar<>", "a bazbar xdsf", []string{"a baz"})
f("<foo>bar<>x", "a bazbar xdsf", []string{"a baz"})
2024-05-19 02:24:32 +00:00
f("foo<bar>", "", []string{""})
f("foo<bar>", "foo", []string{""})
f("foo<bar>", "a foo xdf sdf", []string{" xdf sdf"})
f("foo<bar>", "a foo foobar", []string{" foobar"})
2024-05-19 10:56:08 +00:00
f("foo<bar>baz", "a foo foobar", []string{""})
f("foo<bar>baz", "a foobaz bar", []string{""})
f("foo<bar>baz", "a foo foobar baz", []string{" foobar "})
f("foo<bar>baz", "a foo foobar bazabc", []string{" foobar "})
f("ip=<ip> <> path=<path> ", "x=a, ip=1.2.3.4 method=GET host='abc' path=/foo/bar some tail here", []string{"1.2.3.4", "/foo/bar"})
2024-05-19 13:11:17 +00:00
// escaped pattern
2024-05-19 10:56:08 +00:00
f("ip=&lt;<ip>&gt;", "foo ip=<1.2.3.4> bar", []string{"1.2.3.4"})
f("ip=&lt;<ip>&gt;", "foo ip=<foo&amp;bar> bar", []string{"foo&amp;bar"})
2024-05-19 02:24:32 +00:00
// quoted fields
2024-05-19 10:56:08 +00:00
f(`"msg":<msg>,`, `{"foo":"bar","msg":"foo,b\"ar\n\t","baz":"x"}`, []string{`foo,b"ar` + "\n\t"})
2024-05-19 02:24:32 +00:00
f(`foo=<bar>`, "foo=`bar baz,abc` def", []string{"bar baz,abc"})
2024-05-19 10:56:08 +00:00
f(`foo=<bar> `, "foo=`bar baz,abc` def", []string{"bar baz,abc"})
f(`<foo>`, `"foo,\"bar"`, []string{`foo,"bar`})
f(`<foo>,"bar`, `"foo,\"bar"`, []string{`foo,"bar`})
2024-05-19 02:24:32 +00:00
}
func TestParseExtractFormatStepsSuccess(t *testing.T) {
2024-05-19 10:56:08 +00:00
f := func(s string, stepsExpected []extractFormatStep) {
2024-05-19 02:24:32 +00:00
t.Helper()
steps, err := parseExtractFormatSteps(s)
if err != nil {
t.Fatalf("unexpected error when parsing %q: %s", s, err)
}
if !reflect.DeepEqual(steps, stepsExpected) {
t.Fatalf("unexpected steps for [%s]; got %v; want %v", s, steps, stepsExpected)
}
}
2024-05-19 10:56:08 +00:00
f("<foo>", []extractFormatStep{
2024-05-19 02:24:32 +00:00
{
field: "foo",
},
})
2024-05-19 10:56:08 +00:00
f("<foo>bar", []extractFormatStep{
2024-05-19 02:24:32 +00:00
{
field: "foo",
},
{
prefix: "bar",
},
})
2024-05-19 10:56:08 +00:00
f("<>bar<foo>", []extractFormatStep{
2024-05-19 02:24:32 +00:00
{},
{
prefix: "bar",
2024-05-19 10:56:08 +00:00
field: "foo",
2024-05-19 02:24:32 +00:00
},
})
2024-05-19 10:56:08 +00:00
f("bar<foo>", []extractFormatStep{
2024-05-19 02:24:32 +00:00
{
prefix: "bar",
2024-05-19 10:56:08 +00:00
field: "foo",
2024-05-19 02:24:32 +00:00
},
})
2024-05-19 10:56:08 +00:00
f("bar<foo>abc", []extractFormatStep{
2024-05-19 02:24:32 +00:00
{
prefix: "bar",
2024-05-19 10:56:08 +00:00
field: "foo",
2024-05-19 02:24:32 +00:00
},
{
prefix: "abc",
},
})
2024-05-19 10:56:08 +00:00
f("bar<foo>abc<_>", []extractFormatStep{
2024-05-19 02:24:32 +00:00
{
prefix: "bar",
2024-05-19 10:56:08 +00:00
field: "foo",
2024-05-19 02:24:32 +00:00
},
{
prefix: "abc",
},
})
2024-05-19 10:56:08 +00:00
f("<foo>bar<baz>", []extractFormatStep{
2024-05-19 02:24:32 +00:00
{
field: "foo",
},
{
prefix: "bar",
2024-05-19 10:56:08 +00:00
field: "baz",
2024-05-19 02:24:32 +00:00
},
})
2024-05-19 10:56:08 +00:00
f("bar<foo>baz", []extractFormatStep{
2024-05-19 02:24:32 +00:00
{
prefix: "bar",
2024-05-19 10:56:08 +00:00
field: "foo",
2024-05-19 02:24:32 +00:00
},
{
prefix: "baz",
},
})
2024-05-19 10:56:08 +00:00
f("&lt;<foo>&amp;gt;", []extractFormatStep{
2024-05-19 02:24:32 +00:00
{
prefix: "<",
2024-05-19 10:56:08 +00:00
field: "foo",
2024-05-19 02:24:32 +00:00
},
{
prefix: "&gt;",
},
})
}
func TestParseExtractFormatStepFailure(t *testing.T) {
f := func(s string) {
t.Helper()
_, err := parseExtractFormatSteps(s)
if err == nil {
t.Fatalf("expecting non-nil error when parsing %q", s)
}
}
// empty string
f("")
// zero fields
f("foobar")
2024-05-19 10:56:08 +00:00
// Zero named fields
f("<>")
f("foo<>")
f("<>foo")
f("foo<_>bar<*>baz<>xxx")
2024-05-19 02:24:32 +00:00
// missing delimiter between fields
f("<foo><bar>")
2024-05-19 10:56:08 +00:00
f("<><bar>")
f("<foo><>")
f("bb<foo><><bar>aa")
2024-05-19 02:24:32 +00:00
f("aa<foo><bar>")
f("aa<foo><bar>bb")
// missing >
f("<foo")
f("foo<bar")
}
2024-05-19 13:11:17 +00:00
func TestPipeExtractUpdateNeededFields(t *testing.T) {
f := func(s string, neededFields, unneededFields, neededFieldsExpected, unneededFieldsExpected string) {
t.Helper()
nfs := newTestFieldsSet(neededFields)
unfs := newTestFieldsSet(unneededFields)
lex := newLexer(s)
p, err := parsePipeExtract(lex)
if err != nil {
t.Fatalf("cannot parse %s: %s", s, err)
}
p.updateNeededFields(nfs, unfs)
assertNeededFields(t, nfs, unfs, neededFieldsExpected, unneededFieldsExpected)
}
// all the needed fields
2024-05-19 19:25:52 +00:00
f("extract from x '<foo>'", "*", "", "*", "foo")
2024-05-19 13:11:17 +00:00
// all the needed fields, unneeded fields do not intersect with fromField and output fields
2024-05-19 19:25:52 +00:00
f("extract from x '<foo>'", "*", "f1,f2", "*", "f1,f2,foo")
2024-05-19 13:11:17 +00:00
// all the needed fields, unneeded fields intersect with fromField
2024-05-19 19:25:52 +00:00
f("extract from x '<foo>'", "*", "f2,x", "*", "f2,foo")
2024-05-19 13:11:17 +00:00
// all the needed fields, unneeded fields intersect with output fields
2024-05-19 19:25:52 +00:00
f("extract from x '<foo>x<bar>'", "*", "f2,foo", "*", "bar,f2,foo")
2024-05-19 13:11:17 +00:00
// all the needed fields, unneeded fields intersect with all the output fields
2024-05-19 19:25:52 +00:00
f("extract from x '<foo>x<bar>'", "*", "f2,foo,bar", "*", "bar,f2,foo,x")
2024-05-19 13:11:17 +00:00
// needed fields do not intersect with fromField and output fields
f("extract from x '<foo>x<bar>'", "f1,f2", "", "f1,f2", "")
// needed fields intersect with fromField
f("extract from x '<foo>x<bar>'", "f2,x", "", "f2,x", "")
// needed fields intersect with output fields
f("extract from x '<foo>x<bar>'", "f2,foo", "", "f2,x", "")
// needed fields intersect with fromField and output fields
f("extract from x '<foo>x<bar>'", "f2,foo,x,y", "", "f2,x,y", "")
}