package logstorage import ( "testing" ) func TestParsePipeFormatSuccess(t *testing.T) { f := func(pipeStr string) { t.Helper() expectParsePipeSuccess(t, pipeStr) } f(`format "" as x`) f(`format "<>" as x`) f(`format foo as x`) f(`format "" as _msg`) f(`format "bar" as _msg`) f(`format "barbac" as _msg`) f(`format "barbac" if (x:y) as _msg`) } func TestParsePipeFormatFailure(t *testing.T) { f := func(pipeStr string) { t.Helper() expectParsePipeFailure(t, pipeStr) } f(`format`) f(`format foo`) f(`format foo bar`) f(`format foo as`) f(`format foo if`) f(`format foo as x if (x:y)`) } func TestPipeFormat(t *testing.T) { f := func(pipeStr string, rows, rowsExpected [][]Field) { t.Helper() expectPipeResults(t, pipeStr, rows, rowsExpected) } // plain string into a single field f(`format foo as x`, [][]Field{ { {"_msg", `foobar`}, {"a", "x"}, }, }, [][]Field{ { {"_msg", `foobar`}, {"a", "x"}, {"x", `foo`}, }, }) // plain string with html escaping into a single field f(`format "<foo>" as x`, [][]Field{ { {"_msg", `foobar`}, {"a", "x"}, }, }, [][]Field{ { {"_msg", `foobar`}, {"a", "x"}, {"x", ``}, }, }) // format with empty placeholders into existing field f(`format "<_>foo<_>" as _msg`, [][]Field{ { {"_msg", `foobar`}, {"a", "x"}, }, }, [][]Field{ { {"_msg", `foo`}, {"a", "x"}, }, }) // format with various placeholders into new field f(`format "aaa<_msg>xxx" as x`, [][]Field{ { {"_msg", `foobar`}, {"a", "b"}, }, }, [][]Field{ { {"_msg", `foobar`}, {"a", "b"}, {"x", `aaafoobarxxbx`}, }, }) // format into existing field f(`format "aaa<_msg>xxx" as _msg`, [][]Field{ { {"_msg", `foobar`}, {"a", "b"}, }, }, [][]Field{ { {"_msg", `aaafoobarxxbx`}, {"a", "b"}, }, }) // conditional format over multiple rows f(`format "a: , b: , x: " if (!c:*) as c`, [][]Field{ { {"b", "bar"}, {"a", "foo"}, {"c", "keep-me"}, }, { {"c", ""}, {"a", "f"}, }, { {"b", "x"}, }, }, [][]Field{ { {"b", "bar"}, {"a", "foo"}, {"c", "keep-me"}, }, { {"a", "f"}, {"c", "a: f, b: , x: f"}, }, { {"b", "x"}, {"c", "a: , b: x, x: "}, }, }) } func TestPipeFormatUpdateNeededFields(t *testing.T) { f := func(s string, neededFields, unneededFields, neededFieldsExpected, unneededFieldsExpected string) { t.Helper() expectPipeNeededFields(t, s, neededFields, unneededFields, neededFieldsExpected, unneededFieldsExpected) } // all the needed fields f(`format "foo" as x`, "*", "", "*", "x") f(`format "foo" as x`, "*", "", "*", "x") f(`format "foo" if (f2:z) as x`, "*", "", "*", "x") // unneeded fields do not intersect with pattern and output field f(`format "foo" as x`, "*", "f1,f2", "*", "f1,f2,x") f(`format "foo" as x`, "*", "f1,f2", "*", "f1,f2,x") f(`format "foo" if (f4:z) as x`, "*", "f1,f2", "*", "f1,f2,x") f(`format "foo" if (f1:z) as x`, "*", "f1,f2", "*", "f2,x") // unneeded fields intersect with pattern f(`format "foo" as x`, "*", "f1,f2", "*", "f2,x") f(`format "foo" if (f4:z) as x`, "*", "f1,f2", "*", "f2,x") f(`format "foo" if (f2:z) as x`, "*", "f1,f2", "*", "x") // unneeded fields intersect with output field f(`format "foo" as x`, "*", "x,y", "*", "x,y") f(`format "foo" if (f2:z) as x`, "*", "x,y", "*", "x,y") f(`format "foo" if (y:z) as x`, "*", "x,y", "*", "x,y") // needed fields do not intersect with pattern and output field f(`format "foo" as f2`, "x,y", "", "x,y", "") f(`format "foo" if (f3:z) as f2`, "x,y", "", "x,y", "") f(`format "foo" if (x:z) as f2`, "x,y", "", "x,y", "") // needed fields intersect with pattern field f(`format "foo" as f2`, "f1,y", "", "f1,y", "") f(`format "foo" if (f3:z) as f2`, "f1,y", "", "f1,y", "") f(`format "foo" if (x:z) as f2`, "f1,y", "", "f1,y", "") // needed fields intersect with output field f(`format "foo" as f2`, "f2,y", "", "f1,y", "") f(`format "foo" if (f3:z) as f2`, "f2,y", "", "f1,f3,y", "") f(`format "foo" if (x:z or y:w) as f2`, "f2,y", "", "f1,x,y", "") // needed fields intersect with pattern and output fields f(`format "foo" as f2`, "f1,f2,y", "", "f1,y", "") f(`format "foo" if (f3:z) as f2`, "f1,f2,y", "", "f1,f3,y", "") f(`format "foo" if (x:z or y:w) as f2`, "f1,f2,y", "", "f1,x,y", "") }