mirror of
https://github.com/VictoriaMetrics/VictoriaMetrics.git
synced 2025-01-20 15:16:42 +00:00
wip
This commit is contained in:
parent
01f63b9e94
commit
ae4f92f4cd
6 changed files with 474 additions and 117 deletions
|
@ -3,6 +3,7 @@ package logstorage
|
||||||
import (
|
import (
|
||||||
"math"
|
"math"
|
||||||
"slices"
|
"slices"
|
||||||
|
"strings"
|
||||||
"sync/atomic"
|
"sync/atomic"
|
||||||
"time"
|
"time"
|
||||||
"unsafe"
|
"unsafe"
|
||||||
|
@ -28,12 +29,6 @@ type blockResult struct {
|
||||||
// timestamps contain timestamps for the selected log entries in the block.
|
// timestamps contain timestamps for the selected log entries in the block.
|
||||||
timestamps []int64
|
timestamps []int64
|
||||||
|
|
||||||
// csBufOffset contains csBuf offset for the requested columns.
|
|
||||||
//
|
|
||||||
// columns with indexes below csBufOffset are ignored.
|
|
||||||
// This is needed for simplifying data transformations at pipe stages.
|
|
||||||
csBufOffset int
|
|
||||||
|
|
||||||
// csBuf contains requested columns.
|
// csBuf contains requested columns.
|
||||||
csBuf []blockResultColumn
|
csBuf []blockResultColumn
|
||||||
|
|
||||||
|
@ -52,8 +47,6 @@ func (br *blockResult) reset() {
|
||||||
|
|
||||||
br.timestamps = br.timestamps[:0]
|
br.timestamps = br.timestamps[:0]
|
||||||
|
|
||||||
br.csBufOffset = 0
|
|
||||||
|
|
||||||
clear(br.csBuf)
|
clear(br.csBuf)
|
||||||
br.csBuf = br.csBuf[:0]
|
br.csBuf = br.csBuf[:0]
|
||||||
|
|
||||||
|
@ -1227,56 +1220,67 @@ func (br *blockResult) getBucketedValue(s string, bf *byStatsField) string {
|
||||||
|
|
||||||
// copyColumns copies columns from srcColumnNames to dstColumnNames.
|
// copyColumns copies columns from srcColumnNames to dstColumnNames.
|
||||||
func (br *blockResult) copyColumns(srcColumnNames, dstColumnNames []string) {
|
func (br *blockResult) copyColumns(srcColumnNames, dstColumnNames []string) {
|
||||||
if len(srcColumnNames) == 0 {
|
for i, srcName := range srcColumnNames {
|
||||||
return
|
br.copySingleColumn(srcName, dstColumnNames[i])
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
csBuf := br.csBuf
|
func (br *blockResult) copySingleColumn(srcName, dstName string) {
|
||||||
csBufOffset := len(csBuf)
|
found := false
|
||||||
for _, c := range br.getColumns() {
|
cs := br.getColumns()
|
||||||
if idx := slices.Index(srcColumnNames, c.name); idx >= 0 {
|
csBufLen := len(br.csBuf)
|
||||||
c.name = dstColumnNames[idx]
|
for _, c := range cs {
|
||||||
csBuf = append(csBuf, *c)
|
if c.name != dstName {
|
||||||
// continue is skipped intentionally in order to leave the original column in the columns list.
|
br.csBuf = append(br.csBuf, *c)
|
||||||
}
|
}
|
||||||
if !slices.Contains(dstColumnNames, c.name) {
|
if c.name == srcName {
|
||||||
csBuf = append(csBuf, *c)
|
cCopy := *c
|
||||||
|
cCopy.name = dstName
|
||||||
|
br.csBuf = append(br.csBuf, cCopy)
|
||||||
|
found = true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
br.csBufOffset = csBufOffset
|
if !found {
|
||||||
br.csBuf = csBuf
|
br.addConstColumn(dstName, "")
|
||||||
|
}
|
||||||
|
br.csBuf = append(br.csBuf[:0], br.csBuf[csBufLen:]...)
|
||||||
br.csInitialized = false
|
br.csInitialized = false
|
||||||
|
|
||||||
for _, dstColumnName := range dstColumnNames {
|
|
||||||
br.createMissingColumnByName(dstColumnName)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// renameColumns renames columns from srcColumnNames to dstColumnNames.
|
// renameColumns renames columns from srcColumnNames to dstColumnNames.
|
||||||
func (br *blockResult) renameColumns(srcColumnNames, dstColumnNames []string) {
|
func (br *blockResult) renameColumns(srcColumnNames, dstColumnNames []string) {
|
||||||
if len(srcColumnNames) == 0 {
|
for i, srcName := range srcColumnNames {
|
||||||
return
|
br.renameSingleColumn(srcName, dstColumnNames[i])
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
csBuf := br.csBuf
|
func (br *blockResult) renameSingleColumn(srcName, dstName string) {
|
||||||
csBufOffset := len(csBuf)
|
found := false
|
||||||
for _, c := range br.getColumns() {
|
cs := br.getColumns()
|
||||||
if idx := slices.Index(srcColumnNames, c.name); idx >= 0 {
|
csBufLen := len(br.csBuf)
|
||||||
c.name = dstColumnNames[idx]
|
for _, c := range cs {
|
||||||
csBuf = append(csBuf, *c)
|
if c.name == srcName {
|
||||||
continue
|
cCopy := *c
|
||||||
}
|
cCopy.name = dstName
|
||||||
if !slices.Contains(dstColumnNames, c.name) {
|
br.csBuf = append(br.csBuf, cCopy)
|
||||||
csBuf = append(csBuf, *c)
|
found = true
|
||||||
|
} else if c.name != dstName {
|
||||||
|
br.csBuf = append(br.csBuf, *c)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
br.csBufOffset = csBufOffset
|
if !found {
|
||||||
br.csBuf = csBuf
|
br.addConstColumn(dstName, "")
|
||||||
|
}
|
||||||
|
br.csBuf = append(br.csBuf[:0], br.csBuf[csBufLen:]...)
|
||||||
br.csInitialized = false
|
br.csInitialized = false
|
||||||
|
|
||||||
for _, dstColumnName := range dstColumnNames {
|
|
||||||
br.createMissingColumnByName(dstColumnName)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func debugColumnNames(cs []*blockResultColumn) string {
|
||||||
|
a := make([]string, len(cs))
|
||||||
|
for i, c := range cs {
|
||||||
|
a[i] = c.name
|
||||||
|
}
|
||||||
|
return strings.Join(a, ",")
|
||||||
}
|
}
|
||||||
|
|
||||||
// deleteColumns deletes columns with the given columnNames.
|
// deleteColumns deletes columns with the given columnNames.
|
||||||
|
@ -1285,15 +1289,15 @@ func (br *blockResult) deleteColumns(columnNames []string) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
csBuf := br.csBuf
|
cs := br.getColumns()
|
||||||
csBufOffset := len(csBuf)
|
csBufLen := len(br.csBuf)
|
||||||
for _, c := range br.getColumns() {
|
for _, c := range cs {
|
||||||
if !slices.Contains(columnNames, c.name) {
|
if !slices.Contains(columnNames, c.name) {
|
||||||
csBuf = append(csBuf, *c)
|
br.csBuf = append(br.csBuf, *c)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
br.csBufOffset = csBufOffset
|
|
||||||
br.csBuf = csBuf
|
br.csBuf = append(br.csBuf[:0], br.csBuf[csBufLen:]...)
|
||||||
br.csInitialized = false
|
br.csInitialized = false
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1305,14 +1309,21 @@ func (br *blockResult) setColumns(columnNames []string) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Slow path - construct the requested columns
|
// Slow path - construct the requested columns
|
||||||
csBuf := br.csBuf
|
cs := br.getColumns()
|
||||||
csBufOffset := len(csBuf)
|
csBufLen := len(br.csBuf)
|
||||||
for _, columnName := range columnNames {
|
for _, c := range cs {
|
||||||
c := br.getColumnByName(columnName)
|
if slices.Contains(columnNames, c.name) {
|
||||||
csBuf = append(csBuf, *c)
|
br.csBuf = append(br.csBuf, *c)
|
||||||
}
|
}
|
||||||
br.csBufOffset = csBufOffset
|
}
|
||||||
br.csBuf = csBuf
|
|
||||||
|
for _, columnName := range columnNames {
|
||||||
|
if idx := getBlockResultColumnIdxByName(cs, columnName); idx < 0 {
|
||||||
|
br.addConstColumn(columnName, "")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
br.csBuf = append(br.csBuf[:0], br.csBuf[csBufLen:]...)
|
||||||
br.csInitialized = false
|
br.csInitialized = false
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1344,22 +1355,12 @@ func (br *blockResult) getColumnByName(columnName string) *blockResultColumn {
|
||||||
return &br.csBuf[len(br.csBuf)-1]
|
return &br.csBuf[len(br.csBuf)-1]
|
||||||
}
|
}
|
||||||
|
|
||||||
func (br *blockResult) createMissingColumnByName(columnName string) {
|
|
||||||
for _, c := range br.getColumns() {
|
|
||||||
if c.name == columnName {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
br.addConstColumn(columnName, "")
|
|
||||||
}
|
|
||||||
|
|
||||||
func (br *blockResult) getColumns() []*blockResultColumn {
|
func (br *blockResult) getColumns() []*blockResultColumn {
|
||||||
if br.csInitialized {
|
if br.csInitialized {
|
||||||
return br.cs
|
return br.cs
|
||||||
}
|
}
|
||||||
|
|
||||||
csBuf := br.csBuf[br.csBufOffset:]
|
csBuf := br.csBuf
|
||||||
clear(br.cs)
|
clear(br.cs)
|
||||||
cs := br.cs[:0]
|
cs := br.cs[:0]
|
||||||
for i := range csBuf {
|
for i := range csBuf {
|
||||||
|
|
|
@ -32,29 +32,21 @@ func (pc *pipeCopy) String() string {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (pc *pipeCopy) updateNeededFields(neededFields, unneededFields fieldsSet) {
|
func (pc *pipeCopy) updateNeededFields(neededFields, unneededFields fieldsSet) {
|
||||||
neededSrcFields := make([]bool, len(pc.srcFields))
|
for i := len(pc.srcFields)-1; i >=0 ; i-- {
|
||||||
for i, dstField := range pc.dstFields {
|
srcField := pc.srcFields[i]
|
||||||
if neededFields.contains(dstField) && !unneededFields.contains(dstField) {
|
dstField := pc.dstFields[i]
|
||||||
neededSrcFields[i] = true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if neededFields.contains("*") {
|
if neededFields.contains("*") {
|
||||||
// update only unneeded fields
|
if !unneededFields.contains(dstField) {
|
||||||
unneededFields.addFields(pc.dstFields)
|
unneededFields.add(dstField)
|
||||||
for i, srcField := range pc.srcFields {
|
|
||||||
if neededSrcFields[i] {
|
|
||||||
unneededFields.remove(srcField)
|
unneededFields.remove(srcField)
|
||||||
}
|
}
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
// update only needed fields and reset unneeded fields
|
if neededFields.contains(dstField) {
|
||||||
neededFields.removeFields(pc.dstFields)
|
neededFields.remove(dstField)
|
||||||
for i, srcField := range pc.srcFields {
|
|
||||||
if neededSrcFields[i] {
|
|
||||||
neededFields.add(srcField)
|
neededFields.add(srcField)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
unneededFields.reset()
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -5,6 +5,186 @@ import (
|
||||||
"testing"
|
"testing"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
func TestParsePipeCopySuccess(t *testing.T) {
|
||||||
|
f := func(pipeStr string) {
|
||||||
|
t.Helper()
|
||||||
|
expectParsePipeSuccess(t, pipeStr)
|
||||||
|
}
|
||||||
|
|
||||||
|
f(`copy foo as bar`)
|
||||||
|
f(`copy foo as bar, a as b`)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestParsePipeCopyFailure(t *testing.T) {
|
||||||
|
f := func(pipeStr string) {
|
||||||
|
t.Helper()
|
||||||
|
expectParsePipeFailure(t, pipeStr)
|
||||||
|
}
|
||||||
|
|
||||||
|
f(`copy`)
|
||||||
|
f(`copy x`)
|
||||||
|
f(`copy x as`)
|
||||||
|
f(`copy x y z`)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestPipeCopy(t *testing.T) {
|
||||||
|
f := func(pipeStr string, rows, rowsExpected [][]Field) {
|
||||||
|
t.Helper()
|
||||||
|
expectPipeResults(t, pipeStr, rows, rowsExpected)
|
||||||
|
}
|
||||||
|
|
||||||
|
// single row, copy from existing field
|
||||||
|
f("copy a as b", [][]Field{
|
||||||
|
{
|
||||||
|
{"_msg", `{"foo":"bar"}`},
|
||||||
|
{"a", `test`},
|
||||||
|
},
|
||||||
|
}, [][]Field{
|
||||||
|
{
|
||||||
|
{"_msg", `{"foo":"bar"}`},
|
||||||
|
{"a", `test`},
|
||||||
|
{"b", `test`},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
// single row, copy from existing field to multiple fields
|
||||||
|
f("copy a as b, a as c, _msg as d", [][]Field{
|
||||||
|
{
|
||||||
|
{"_msg", `{"foo":"bar"}`},
|
||||||
|
{"a", `test`},
|
||||||
|
},
|
||||||
|
}, [][]Field{
|
||||||
|
{
|
||||||
|
{"_msg", `{"foo":"bar"}`},
|
||||||
|
{"a", `test`},
|
||||||
|
{"b", `test`},
|
||||||
|
{"c", `test`},
|
||||||
|
{"d", `{"foo":"bar"}`},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
// single row, copy from non-exsiting field
|
||||||
|
f("copy x as b", [][]Field{
|
||||||
|
{
|
||||||
|
{"_msg", `{"foo":"bar"}`},
|
||||||
|
{"a", `test`},
|
||||||
|
},
|
||||||
|
}, [][]Field{
|
||||||
|
{
|
||||||
|
{"_msg", `{"foo":"bar"}`},
|
||||||
|
{"a", `test`},
|
||||||
|
{"b", ``},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
// copy to existing field
|
||||||
|
f("copy _msg as a", [][]Field{
|
||||||
|
{
|
||||||
|
{"_msg", `{"foo":"bar"}`},
|
||||||
|
{"a", `test`},
|
||||||
|
},
|
||||||
|
}, [][]Field{
|
||||||
|
{
|
||||||
|
{"_msg", `{"foo":"bar"}`},
|
||||||
|
{"a", `{"foo":"bar"}`},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
// copy to itself
|
||||||
|
f("copy a as a", [][]Field{
|
||||||
|
{
|
||||||
|
{"_msg", `{"foo":"bar"}`},
|
||||||
|
{"a", `test`},
|
||||||
|
},
|
||||||
|
}, [][]Field{
|
||||||
|
{
|
||||||
|
{"_msg", `{"foo":"bar"}`},
|
||||||
|
{"a", `test`},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
// swap copy
|
||||||
|
f("copy a as b, _msg as a, b as _msg", [][]Field{
|
||||||
|
{
|
||||||
|
{"_msg", `{"foo":"bar"}`},
|
||||||
|
{"a", `test`},
|
||||||
|
},
|
||||||
|
}, [][]Field{
|
||||||
|
{
|
||||||
|
{"_msg", `test`},
|
||||||
|
{"a", `{"foo":"bar"}`},
|
||||||
|
{"b", `test`},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
// copy to the same field multiple times
|
||||||
|
f("copy a as b, _msg as b", [][]Field{
|
||||||
|
{
|
||||||
|
{"_msg", `{"foo":"bar"}`},
|
||||||
|
{"a", `test`},
|
||||||
|
},
|
||||||
|
}, [][]Field{
|
||||||
|
{
|
||||||
|
{"_msg", `{"foo":"bar"}`},
|
||||||
|
{"a", `test`},
|
||||||
|
{"b", `{"foo":"bar"}`},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
// chain copy
|
||||||
|
f("copy a as b, b as c", [][]Field{
|
||||||
|
{
|
||||||
|
{"_msg", `{"foo":"bar"}`},
|
||||||
|
{"a", `test`},
|
||||||
|
},
|
||||||
|
}, [][]Field{
|
||||||
|
{
|
||||||
|
{"_msg", `{"foo":"bar"}`},
|
||||||
|
{"a", `test`},
|
||||||
|
{"b", `test`},
|
||||||
|
{"c", `test`},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
// Multiple rows
|
||||||
|
f("copy a as b", [][]Field{
|
||||||
|
{
|
||||||
|
{"_msg", `{"foo":"bar"}`},
|
||||||
|
{"a", `test`},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
{"a", `foobar`},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
{"b", `baz`},
|
||||||
|
{"c", "d"},
|
||||||
|
{"e", "afdf"},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
{"c", "dss"},
|
||||||
|
},
|
||||||
|
}, [][]Field{
|
||||||
|
{
|
||||||
|
{"_msg", `{"foo":"bar"}`},
|
||||||
|
{"a", `test`},
|
||||||
|
{"b", `test`},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
{"a", `foobar`},
|
||||||
|
{"b", `foobar`},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
{"b", ``},
|
||||||
|
{"c", "d"},
|
||||||
|
{"e", "afdf"},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
{"c", "dss"},
|
||||||
|
{"b", ""},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
func TestPipeCopyUpdateNeededFields(t *testing.T) {
|
func TestPipeCopyUpdateNeededFields(t *testing.T) {
|
||||||
f := func(s, neededFields, unneededFields, neededFieldsExpected, unneededFieldsExpected string) {
|
f := func(s, neededFields, unneededFields, neededFieldsExpected, unneededFieldsExpected string) {
|
||||||
t.Helper()
|
t.Helper()
|
||||||
|
@ -13,6 +193,7 @@ func TestPipeCopyUpdateNeededFields(t *testing.T) {
|
||||||
|
|
||||||
// all the needed fields
|
// all the needed fields
|
||||||
f("copy s1 d1, s2 d2", "*", "", "*", "d1,d2")
|
f("copy s1 d1, s2 d2", "*", "", "*", "d1,d2")
|
||||||
|
f("copy a a", "*", "", "*", "")
|
||||||
|
|
||||||
// all the needed fields, unneeded fields do not intersect with src and dst
|
// all the needed fields, unneeded fields do not intersect with src and dst
|
||||||
f("copy s1 d1 ,s2 d2", "*", "f1,f2", "*", "d1,d2,f1,f2")
|
f("copy s1 d1 ,s2 d2", "*", "f1,f2", "*", "d1,d2,f1,f2")
|
||||||
|
|
|
@ -32,33 +32,25 @@ func (pr *pipeRename) String() string {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (pr *pipeRename) updateNeededFields(neededFields, unneededFields fieldsSet) {
|
func (pr *pipeRename) updateNeededFields(neededFields, unneededFields fieldsSet) {
|
||||||
neededSrcFields := make([]bool, len(pr.srcFields))
|
for i := len(pr.srcFields)-1; i >=0 ; i-- {
|
||||||
for i, dstField := range pr.dstFields {
|
srcField := pr.srcFields[i]
|
||||||
if neededFields.contains(dstField) && !unneededFields.contains(dstField) {
|
dstField := pr.dstFields[i]
|
||||||
neededSrcFields[i] = true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if neededFields.contains("*") {
|
if neededFields.contains("*") {
|
||||||
// update only unneeded fields
|
if unneededFields.contains(dstField) {
|
||||||
unneededFields.addFields(pr.dstFields)
|
|
||||||
for i, srcField := range pr.srcFields {
|
|
||||||
if neededSrcFields[i] {
|
|
||||||
unneededFields.remove(srcField)
|
|
||||||
} else {
|
|
||||||
unneededFields.add(srcField)
|
unneededFields.add(srcField)
|
||||||
}
|
} else {
|
||||||
|
unneededFields.add(dstField)
|
||||||
|
unneededFields.remove(srcField)
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// update only needed fields and reset unneeded fields
|
if neededFields.contains(dstField) {
|
||||||
neededFields.removeFields(pr.dstFields)
|
neededFields.remove(dstField)
|
||||||
for i, srcField := range pr.srcFields {
|
|
||||||
if neededSrcFields[i] {
|
|
||||||
neededFields.add(srcField)
|
neededFields.add(srcField)
|
||||||
} else {
|
} else {
|
||||||
neededFields.remove(srcField)
|
neededFields.remove(srcField)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
unneededFields.reset()
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -4,6 +4,175 @@ import (
|
||||||
"testing"
|
"testing"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
func TestParsePipeRenameSuccess(t *testing.T) {
|
||||||
|
f := func(pipeStr string) {
|
||||||
|
t.Helper()
|
||||||
|
expectParsePipeSuccess(t, pipeStr)
|
||||||
|
}
|
||||||
|
|
||||||
|
f(`rename foo as bar`)
|
||||||
|
f(`rename foo as bar, a as b`)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestParsePipeRenameFailure(t *testing.T) {
|
||||||
|
f := func(pipeStr string) {
|
||||||
|
t.Helper()
|
||||||
|
expectParsePipeFailure(t, pipeStr)
|
||||||
|
}
|
||||||
|
|
||||||
|
f(`rename`)
|
||||||
|
f(`rename x`)
|
||||||
|
f(`rename x as`)
|
||||||
|
f(`rename x y z`)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestPipeRename(t *testing.T) {
|
||||||
|
f := func(pipeStr string, rows, rowsExpected [][]Field) {
|
||||||
|
t.Helper()
|
||||||
|
expectPipeResults(t, pipeStr, rows, rowsExpected)
|
||||||
|
}
|
||||||
|
|
||||||
|
// single row, rename from existing field
|
||||||
|
f("rename a as b", [][]Field{
|
||||||
|
{
|
||||||
|
{"_msg", `{"foo":"bar"}`},
|
||||||
|
{"a", `test`},
|
||||||
|
},
|
||||||
|
}, [][]Field{
|
||||||
|
{
|
||||||
|
{"_msg", `{"foo":"bar"}`},
|
||||||
|
{"b", `test`},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
// single row, rename from existing field to multiple fields
|
||||||
|
f("rename a as b, a as c, _msg as d", [][]Field{
|
||||||
|
{
|
||||||
|
{"_msg", `{"foo":"bar"}`},
|
||||||
|
{"a", `test`},
|
||||||
|
},
|
||||||
|
}, [][]Field{
|
||||||
|
{
|
||||||
|
{"b", `test`},
|
||||||
|
{"c", ``},
|
||||||
|
{"d", `{"foo":"bar"}`},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
// single row, rename from non-exsiting field
|
||||||
|
f("rename x as b", [][]Field{
|
||||||
|
{
|
||||||
|
{"_msg", `{"foo":"bar"}`},
|
||||||
|
{"a", `test`},
|
||||||
|
},
|
||||||
|
}, [][]Field{
|
||||||
|
{
|
||||||
|
{"_msg", `{"foo":"bar"}`},
|
||||||
|
{"a", `test`},
|
||||||
|
{"b", ``},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
// rename to existing field
|
||||||
|
f("rename _msg as a", [][]Field{
|
||||||
|
{
|
||||||
|
{"_msg", `{"foo":"bar"}`},
|
||||||
|
{"a", `test`},
|
||||||
|
},
|
||||||
|
}, [][]Field{
|
||||||
|
{
|
||||||
|
{"a", `{"foo":"bar"}`},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
// rename to itself
|
||||||
|
f("rename a as a", [][]Field{
|
||||||
|
{
|
||||||
|
{"_msg", `{"foo":"bar"}`},
|
||||||
|
{"a", `test`},
|
||||||
|
},
|
||||||
|
}, [][]Field{
|
||||||
|
{
|
||||||
|
{"_msg", `{"foo":"bar"}`},
|
||||||
|
{"a", `test`},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
// swap rename
|
||||||
|
f("rename a as b, _msg as a, b as _msg", [][]Field{
|
||||||
|
{
|
||||||
|
{"_msg", `{"foo":"bar"}`},
|
||||||
|
{"a", `test`},
|
||||||
|
},
|
||||||
|
}, [][]Field{
|
||||||
|
{
|
||||||
|
{"_msg", `test`},
|
||||||
|
{"a", `{"foo":"bar"}`},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
// rename to the same field multiple times
|
||||||
|
f("rename a as b, _msg as b", [][]Field{
|
||||||
|
{
|
||||||
|
{"_msg", `{"foo":"bar"}`},
|
||||||
|
{"a", `test`},
|
||||||
|
},
|
||||||
|
}, [][]Field{
|
||||||
|
{
|
||||||
|
{"b", `{"foo":"bar"}`},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
// chain rename (shouldn't work - otherwise swap rename will break)
|
||||||
|
f("rename a as b, b as c", [][]Field{
|
||||||
|
{
|
||||||
|
{"_msg", `{"foo":"bar"}`},
|
||||||
|
{"a", `test`},
|
||||||
|
},
|
||||||
|
}, [][]Field{
|
||||||
|
{
|
||||||
|
{"_msg", `{"foo":"bar"}`},
|
||||||
|
{"c", `test`},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
// Multiple rows
|
||||||
|
f("rename a as b", [][]Field{
|
||||||
|
{
|
||||||
|
{"_msg", `{"foo":"bar"}`},
|
||||||
|
{"a", `test`},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
{"a", `foobar`},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
{"b", `baz`},
|
||||||
|
{"c", "d"},
|
||||||
|
{"e", "afdf"},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
{"c", "dss"},
|
||||||
|
},
|
||||||
|
}, [][]Field{
|
||||||
|
{
|
||||||
|
{"_msg", `{"foo":"bar"}`},
|
||||||
|
{"b", `test`},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
{"b", `foobar`},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
{"b", ``},
|
||||||
|
{"c", "d"},
|
||||||
|
{"e", "afdf"},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
{"c", "dss"},
|
||||||
|
{"b", ""},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
func TestPipeRenameUpdateNeededFields(t *testing.T) {
|
func TestPipeRenameUpdateNeededFields(t *testing.T) {
|
||||||
f := func(s, neededFields, unneededFields, neededFieldsExpected, unneededFieldsExpected string) {
|
f := func(s, neededFields, unneededFields, neededFieldsExpected, unneededFieldsExpected string) {
|
||||||
t.Helper()
|
t.Helper()
|
||||||
|
@ -12,6 +181,7 @@ func TestPipeRenameUpdateNeededFields(t *testing.T) {
|
||||||
|
|
||||||
// all the needed fields
|
// all the needed fields
|
||||||
f("rename s1 d1, s2 d2", "*", "", "*", "d1,d2")
|
f("rename s1 d1, s2 d2", "*", "", "*", "d1,d2")
|
||||||
|
f("rename a a", "*", "", "*", "")
|
||||||
|
|
||||||
// all the needed fields, unneeded fields do not intersect with src and dst
|
// all the needed fields, unneeded fields do not intersect with src and dst
|
||||||
f("rename s1 d1, s2 d2", "*", "f1,f2", "*", "d1,d2,f1,f2")
|
f("rename s1 d1, s2 d2", "*", "f1,f2", "*", "d1,d2,f1,f2")
|
||||||
|
|
|
@ -341,35 +341,56 @@ func (pp *testPipeProcessor) expectRows(t *testing.T, expectedRows [][]Field) {
|
||||||
}
|
}
|
||||||
|
|
||||||
func sortTestRows(rows [][]Field) {
|
func sortTestRows(rows [][]Field) {
|
||||||
|
for _, row := range rows {
|
||||||
|
sortTestFields(row)
|
||||||
|
}
|
||||||
slices.SortFunc(rows, func(a, b []Field) int {
|
slices.SortFunc(rows, func(a, b []Field) int {
|
||||||
reverse := -1
|
reverse := false
|
||||||
if len(a) > len(b) {
|
if len(a) > len(b) {
|
||||||
reverse = 1
|
reverse = true
|
||||||
a, b = b, a
|
a, b = b, a
|
||||||
}
|
}
|
||||||
for i, fA := range a {
|
for i, fA := range a {
|
||||||
fB := b[i]
|
fB := b[i]
|
||||||
if fA.Name == fB.Name {
|
result := cmpTestFields(fA, fB)
|
||||||
if fA.Value == fB.Value {
|
if result == 0 {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
if fA.Value < fB.Value {
|
if reverse {
|
||||||
return reverse
|
result = -result
|
||||||
}
|
}
|
||||||
return -reverse
|
return result
|
||||||
}
|
|
||||||
if fA.Name < fB.Name {
|
|
||||||
return reverse
|
|
||||||
}
|
|
||||||
return -reverse
|
|
||||||
}
|
}
|
||||||
if len(a) == len(b) {
|
if len(a) == len(b) {
|
||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
return reverse
|
if reverse {
|
||||||
|
return 1
|
||||||
|
}
|
||||||
|
return -1
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func sortTestFields(fields []Field) {
|
||||||
|
slices.SortFunc(fields, cmpTestFields)
|
||||||
|
}
|
||||||
|
|
||||||
|
func cmpTestFields(a, b Field) int {
|
||||||
|
if a.Name == b.Name {
|
||||||
|
if a.Value == b.Value {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
if a.Value < b.Value {
|
||||||
|
return -1
|
||||||
|
}
|
||||||
|
return 1
|
||||||
|
}
|
||||||
|
if a.Name < b.Name {
|
||||||
|
return -1
|
||||||
|
}
|
||||||
|
return 1
|
||||||
|
}
|
||||||
|
|
||||||
func rowsToString(rows [][]Field) string {
|
func rowsToString(rows [][]Field) string {
|
||||||
a := make([]string, len(rows))
|
a := make([]string, len(rows))
|
||||||
for i, row := range rows {
|
for i, row := range rows {
|
||||||
|
|
Loading…
Reference in a new issue