mirror of
https://github.com/VictoriaMetrics/VictoriaMetrics.git
synced 2025-01-20 15:16:42 +00:00
wip
This commit is contained in:
parent
eac0722068
commit
c5734e18b9
9 changed files with 164 additions and 34 deletions
|
@ -3,7 +3,6 @@ package logstorage
|
||||||
import (
|
import (
|
||||||
"math"
|
"math"
|
||||||
"slices"
|
"slices"
|
||||||
"strings"
|
|
||||||
"sync/atomic"
|
"sync/atomic"
|
||||||
"time"
|
"time"
|
||||||
"unsafe"
|
"unsafe"
|
||||||
|
@ -201,19 +200,6 @@ func (br *blockResult) sizeBytes() int {
|
||||||
return n
|
return n
|
||||||
}
|
}
|
||||||
|
|
||||||
// addResultColumns adds the given rcs to br.
|
|
||||||
//
|
|
||||||
// The br is valid only until rcs are modified.
|
|
||||||
func (br *blockResult) addResultColumns(rcs []resultColumn) {
|
|
||||||
if len(rcs) == 0 || len(rcs[0].values) == 0 {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
for i := range rcs {
|
|
||||||
br.addResultColumn(&rcs[i])
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// setResultColumns sets the given rcs as br columns.
|
// setResultColumns sets the given rcs as br columns.
|
||||||
//
|
//
|
||||||
// The br is valid only until rcs are modified.
|
// The br is valid only until rcs are modified.
|
||||||
|
@ -1275,14 +1261,6 @@ func (br *blockResult) renameSingleColumn(srcName, dstName string) {
|
||||||
br.csInitialized = false
|
br.csInitialized = false
|
||||||
}
|
}
|
||||||
|
|
||||||
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.
|
||||||
func (br *blockResult) deleteColumns(columnNames []string) {
|
func (br *blockResult) deleteColumns(columnNames []string) {
|
||||||
if len(columnNames) == 0 {
|
if len(columnNames) == 0 {
|
||||||
|
@ -1811,6 +1789,11 @@ type resultColumn struct {
|
||||||
values []string
|
values []string
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (rc *resultColumn) reset() {
|
||||||
|
rc.name = ""
|
||||||
|
rc.resetValues()
|
||||||
|
}
|
||||||
|
|
||||||
func (rc *resultColumn) resetValues() {
|
func (rc *resultColumn) resetValues() {
|
||||||
clear(rc.values)
|
clear(rc.values)
|
||||||
rc.values = rc.values[:0]
|
rc.values = rc.values[:0]
|
||||||
|
@ -1819,8 +1802,8 @@ func (rc *resultColumn) resetValues() {
|
||||||
func appendResultColumnWithName(dst []resultColumn, name string) []resultColumn {
|
func appendResultColumnWithName(dst []resultColumn, name string) []resultColumn {
|
||||||
dst = slicesutil.SetLength(dst, len(dst)+1)
|
dst = slicesutil.SetLength(dst, len(dst)+1)
|
||||||
rc := &dst[len(dst)-1]
|
rc := &dst[len(dst)-1]
|
||||||
rc.resetValues()
|
|
||||||
rc.name = name
|
rc.name = name
|
||||||
|
rc.resetValues()
|
||||||
return dst
|
return dst
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -60,7 +60,9 @@ func TestPipeDelete(t *testing.T) {
|
||||||
{"_msg", `{"foo":"bar"}`},
|
{"_msg", `{"foo":"bar"}`},
|
||||||
{"a", `test`},
|
{"a", `test`},
|
||||||
},
|
},
|
||||||
}, [][]Field{})
|
}, [][]Field{
|
||||||
|
{},
|
||||||
|
})
|
||||||
|
|
||||||
// delete non-existing fields
|
// delete non-existing fields
|
||||||
f("delete foo, _msg, bar", [][]Field{
|
f("delete foo, _msg, bar", [][]Field{
|
||||||
|
@ -93,6 +95,8 @@ func TestPipeDelete(t *testing.T) {
|
||||||
{"b", "df"},
|
{"b", "df"},
|
||||||
},
|
},
|
||||||
}, [][]Field{
|
}, [][]Field{
|
||||||
|
{},
|
||||||
|
{},
|
||||||
{
|
{
|
||||||
{"b", `baz`},
|
{"b", `baz`},
|
||||||
{"c", "d"},
|
{"c", "d"},
|
||||||
|
|
|
@ -68,10 +68,9 @@ func (pe *pipeExtract) updateNeededFields(neededFields, unneededFields fieldsSet
|
||||||
func (pe *pipeExtract) newPipeProcessor(workersCount int, _ <-chan struct{}, _ func(), ppBase pipeProcessor) pipeProcessor {
|
func (pe *pipeExtract) newPipeProcessor(workersCount int, _ <-chan struct{}, _ func(), ppBase pipeProcessor) pipeProcessor {
|
||||||
shards := make([]pipeExtractProcessorShard, workersCount)
|
shards := make([]pipeExtractProcessorShard, workersCount)
|
||||||
for i := range shards {
|
for i := range shards {
|
||||||
ef := newPattern(pe.steps)
|
|
||||||
shards[i] = pipeExtractProcessorShard{
|
shards[i] = pipeExtractProcessorShard{
|
||||||
pipeExtractProcessorShardNopad: pipeExtractProcessorShardNopad{
|
pipeExtractProcessorShardNopad: pipeExtractProcessorShardNopad{
|
||||||
ef: ef,
|
ef: newPattern(pe.steps),
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -516,7 +516,7 @@ func (wctx *pipeSortWriteContext) writeNextRow(shard *pipeSortProcessorShard) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if !areEqualColumns {
|
if !areEqualColumns {
|
||||||
// send the current block to bbBase and construct a block with new set of columns
|
// send the current block to ppBase and construct a block with new set of columns
|
||||||
wctx.flush()
|
wctx.flush()
|
||||||
|
|
||||||
rcs = wctx.rcs[:0]
|
rcs = wctx.rcs[:0]
|
||||||
|
|
|
@ -154,6 +154,28 @@ func TestPipeSort(t *testing.T) {
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
|
||||||
|
// Sort by multiple fields with limit desc
|
||||||
|
f("sort by (a, b) desc limit 1", [][]Field{
|
||||||
|
{
|
||||||
|
{"_msg", `abc`},
|
||||||
|
{"a", `2`},
|
||||||
|
{"b", `3`},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
{"_msg", `def`},
|
||||||
|
{"a", `1`},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
{"a", `2`},
|
||||||
|
{"b", `54`},
|
||||||
|
},
|
||||||
|
}, [][]Field{
|
||||||
|
{
|
||||||
|
{"a", `2`},
|
||||||
|
{"b", `54`},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
// Sort by multiple fields with offset
|
// Sort by multiple fields with offset
|
||||||
f("sort by (a, b) offset 1", [][]Field{
|
f("sort by (a, b) offset 1", [][]Field{
|
||||||
{
|
{
|
||||||
|
@ -203,6 +225,28 @@ func TestPipeSort(t *testing.T) {
|
||||||
{"b", `3`},
|
{"b", `3`},
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
|
||||||
|
// Sort by multiple fields with offset and limit
|
||||||
|
f("sort by (a, b) desc offset 2 limit 100", [][]Field{
|
||||||
|
{
|
||||||
|
{"_msg", `abc`},
|
||||||
|
{"a", `2`},
|
||||||
|
{"b", `3`},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
{"_msg", `def`},
|
||||||
|
{"a", `1`},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
{"a", `2`},
|
||||||
|
{"b", `54`},
|
||||||
|
},
|
||||||
|
}, [][]Field{
|
||||||
|
{
|
||||||
|
{"_msg", `def`},
|
||||||
|
{"a", `1`},
|
||||||
|
},
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestPipeSortUpdateNeededFields(t *testing.T) {
|
func TestPipeSortUpdateNeededFields(t *testing.T) {
|
||||||
|
|
|
@ -4,6 +4,85 @@ import (
|
||||||
"testing"
|
"testing"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
func TestParsePipeStatsSuccess(t *testing.T) {
|
||||||
|
f := func(pipeStr string) {
|
||||||
|
t.Helper()
|
||||||
|
expectParsePipeSuccess(t, pipeStr)
|
||||||
|
}
|
||||||
|
|
||||||
|
f(`stats count(*) as rows`)
|
||||||
|
f(`stats by (x) count(*) as rows, count_uniq(x) as uniqs`)
|
||||||
|
f(`stats by (_time:month offset 6.5h, y) count(*) as rows, count_uniq(x) as uniqs`)
|
||||||
|
f(`stats by (_time:month offset 6.5h, y) count(*) if (q:w) as rows, count_uniq(x) as uniqs`)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestParsePipeStatsFailure(t *testing.T) {
|
||||||
|
f := func(pipeStr string) {
|
||||||
|
t.Helper()
|
||||||
|
expectParsePipeFailure(t, pipeStr)
|
||||||
|
}
|
||||||
|
|
||||||
|
f(`stats`)
|
||||||
|
f(`stats by`)
|
||||||
|
f(`stats foo`)
|
||||||
|
f(`stats count`)
|
||||||
|
f(`stats if (x:y)`)
|
||||||
|
f(`stats by(x) foo`)
|
||||||
|
f(`stats by(x:abc) count() rows`)
|
||||||
|
f(`stats by(x:1h offset) count () rows`)
|
||||||
|
f(`stats by(x:1h offset foo) count() rows`)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestPipeStats(t *testing.T) {
|
||||||
|
f := func(pipeStr string, rows, rowsExpected [][]Field) {
|
||||||
|
t.Helper()
|
||||||
|
expectPipeResults(t, pipeStr, rows, rowsExpected)
|
||||||
|
}
|
||||||
|
|
||||||
|
f("stats count(*) as rows", [][]Field{
|
||||||
|
{
|
||||||
|
{"_msg", `abc`},
|
||||||
|
{"a", `2`},
|
||||||
|
{"b", `3`},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
{"_msg", `def`},
|
||||||
|
{"a", `1`},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
{"a", `2`},
|
||||||
|
{"b", `54`},
|
||||||
|
},
|
||||||
|
}, [][]Field{
|
||||||
|
{
|
||||||
|
{"rows", "3"},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
f("stats by (a) count(*) as rows", [][]Field{
|
||||||
|
{
|
||||||
|
{"_msg", `abc`},
|
||||||
|
{"a", `2`},
|
||||||
|
{"b", `3`},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
{"_msg", `def`},
|
||||||
|
{"a", `1`},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
{"a", `2`},
|
||||||
|
{"b", `54`},
|
||||||
|
},
|
||||||
|
}, [][]Field{
|
||||||
|
{
|
||||||
|
{"a", "1"},
|
||||||
|
{"rows", "1"},
|
||||||
|
{"a", "2"},
|
||||||
|
{"rows", "2"},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
func TestPipeStatsUpdateNeededFields(t *testing.T) {
|
func TestPipeStatsUpdateNeededFields(t *testing.T) {
|
||||||
f := func(s, neededFields, unneededFields, neededFieldsExpected, unneededFieldsExpected string) {
|
f := func(s, neededFields, unneededFields, neededFieldsExpected, unneededFieldsExpected string) {
|
||||||
t.Helper()
|
t.Helper()
|
||||||
|
|
|
@ -457,7 +457,7 @@ func (wctx *pipeTopkWriteContext) writeNextRow(shard *pipeTopkProcessorShard) bo
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if !areEqualColumns {
|
if !areEqualColumns {
|
||||||
// send the current block to bbBase and construct a block with new set of columns
|
// send the current block to ppBase and construct a block with new set of columns
|
||||||
wctx.flush()
|
wctx.flush()
|
||||||
|
|
||||||
rcs = wctx.rcs[:0]
|
rcs = wctx.rcs[:0]
|
||||||
|
|
|
@ -354,7 +354,7 @@ func (wctx *pipeUniqWriteContext) writeRow(rowFields []Field) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if !areEqualColumns {
|
if !areEqualColumns {
|
||||||
// send the current block to bbBase and construct a block with new set of columns
|
// send the current block to ppBase and construct a block with new set of columns
|
||||||
wctx.flush()
|
wctx.flush()
|
||||||
|
|
||||||
rcs = wctx.rcs[:0]
|
rcs = wctx.rcs[:0]
|
||||||
|
|
|
@ -22,10 +22,15 @@ func (uctx *fieldsUnpackerContext) resetFields() {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (uctx *fieldsUnpackerContext) addField(name, value, fieldPrefix string) {
|
func (uctx *fieldsUnpackerContext) addField(name, value, fieldPrefix string) {
|
||||||
nameBuf := uctx.a.newBytes(len(fieldPrefix) + len(name))
|
nameCopy := ""
|
||||||
copy(nameBuf, fieldPrefix)
|
if fieldPrefix != "" {
|
||||||
copy(nameBuf[len(fieldPrefix):], name)
|
nameBuf := uctx.a.newBytes(len(fieldPrefix) + len(name))
|
||||||
nameCopy := bytesutil.ToUnsafeString(nameBuf)
|
copy(nameBuf, fieldPrefix)
|
||||||
|
copy(nameBuf[len(fieldPrefix):], name)
|
||||||
|
nameCopy = bytesutil.ToUnsafeString(nameBuf)
|
||||||
|
} else {
|
||||||
|
nameCopy = uctx.a.copyString(name)
|
||||||
|
}
|
||||||
|
|
||||||
valueCopy := uctx.a.copyString(value)
|
valueCopy := uctx.a.copyString(value)
|
||||||
|
|
||||||
|
@ -115,7 +120,23 @@ type pipeUnpackWriteContext struct {
|
||||||
valuesLen int
|
valuesLen int
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (wctx *pipeUnpackWriteContext) reset() {
|
||||||
|
wctx.brSrc = nil
|
||||||
|
wctx.csSrc = nil
|
||||||
|
wctx.ppBase = nil
|
||||||
|
|
||||||
|
rcs := wctx.rcs
|
||||||
|
for i := range rcs {
|
||||||
|
rcs[i].reset()
|
||||||
|
}
|
||||||
|
wctx.rcs = rcs[:0]
|
||||||
|
|
||||||
|
wctx.valuesLen = 0
|
||||||
|
}
|
||||||
|
|
||||||
func (wctx *pipeUnpackWriteContext) init(brSrc *blockResult, ppBase pipeProcessor) {
|
func (wctx *pipeUnpackWriteContext) init(brSrc *blockResult, ppBase pipeProcessor) {
|
||||||
|
wctx.reset()
|
||||||
|
|
||||||
wctx.brSrc = brSrc
|
wctx.brSrc = brSrc
|
||||||
wctx.csSrc = brSrc.getColumns()
|
wctx.csSrc = brSrc.getColumns()
|
||||||
wctx.ppBase = ppBase
|
wctx.ppBase = ppBase
|
||||||
|
@ -135,7 +156,7 @@ func (wctx *pipeUnpackWriteContext) writeRow(rowIdx int, extraFields []Field) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if !areEqualColumns {
|
if !areEqualColumns {
|
||||||
// send the current block to bbBase and construct a block with new set of columns
|
// send the current block to ppBase and construct a block with new set of columns
|
||||||
wctx.flush()
|
wctx.flush()
|
||||||
|
|
||||||
rcs = wctx.rcs[:0]
|
rcs = wctx.rcs[:0]
|
||||||
|
|
Loading…
Reference in a new issue