mirror of
https://github.com/VictoriaMetrics/VictoriaMetrics.git
synced 2024-12-31 15:06:26 +00:00
wip
This commit is contained in:
parent
33610de341
commit
46bc1c3435
7 changed files with 139 additions and 17 deletions
|
@ -21,6 +21,7 @@ according to [these docs](https://docs.victoriametrics.com/victorialogs/quicksta
|
||||||
|
|
||||||
* FEATURE: add [`replace_regexp` pipe](https://docs.victoriametrics.com/victorialogs/logsql/#replace_regexp-pipe), which allows updating [log fields](https://docs.victoriametrics.com/victorialogs/keyconcepts/#data-model) with regular expressions.
|
* FEATURE: add [`replace_regexp` pipe](https://docs.victoriametrics.com/victorialogs/logsql/#replace_regexp-pipe), which allows updating [log fields](https://docs.victoriametrics.com/victorialogs/keyconcepts/#data-model) with regular expressions.
|
||||||
|
|
||||||
|
* BUGFIX: do not return referenced fields if they weren't present in the original logs. For example, `_time:5m | format if (non_existing_field:"") "abc"` could return empty `non_exiting_field`, while it shuldn't be returned because it is missing in the original logs.
|
||||||
* BUGFIX: properly initialize values for [`in(...)` filter](https://docs.victoriametrics.com/victorialogs/logsql/#exact-filter) inside [`filter` pipe](https://docs.victoriametrics.com/victorialogs/logsql/#filter-pipe) if the `in(...)` contains other [filters](https://docs.victoriametrics.com/victorialogs/logsql/#filters). For example, `_time:5m | filter ip:in(user_type:admin | fields ip)` now works correctly.
|
* BUGFIX: properly initialize values for [`in(...)` filter](https://docs.victoriametrics.com/victorialogs/logsql/#exact-filter) inside [`filter` pipe](https://docs.victoriametrics.com/victorialogs/logsql/#filter-pipe) if the `in(...)` contains other [filters](https://docs.victoriametrics.com/victorialogs/logsql/#filters). For example, `_time:5m | filter ip:in(user_type:admin | fields ip)` now works correctly.
|
||||||
|
|
||||||
## [v0.11.0](https://github.com/VictoriaMetrics/VictoriaMetrics/releases/tag/v0.11.0-victorialogs)
|
## [v0.11.0](https://github.com/VictoriaMetrics/VictoriaMetrics/releases/tag/v0.11.0-victorialogs)
|
||||||
|
|
|
@ -31,6 +31,9 @@ type blockResult struct {
|
||||||
// csBuf contains requested columns.
|
// csBuf contains requested columns.
|
||||||
csBuf []blockResultColumn
|
csBuf []blockResultColumn
|
||||||
|
|
||||||
|
// csEmpty contains non-existing columns, which were referenced via getColumnByName()
|
||||||
|
csEmpty []blockResultColumn
|
||||||
|
|
||||||
// cs contains cached pointers to requested columns returned from getColumns() if csInitialized=true.
|
// cs contains cached pointers to requested columns returned from getColumns() if csInitialized=true.
|
||||||
cs []*blockResultColumn
|
cs []*blockResultColumn
|
||||||
|
|
||||||
|
@ -49,6 +52,9 @@ func (br *blockResult) reset() {
|
||||||
clear(br.csBuf)
|
clear(br.csBuf)
|
||||||
br.csBuf = br.csBuf[:0]
|
br.csBuf = br.csBuf[:0]
|
||||||
|
|
||||||
|
clear(br.csEmpty)
|
||||||
|
br.csEmpty = br.csEmpty[:0]
|
||||||
|
|
||||||
clear(br.cs)
|
clear(br.cs)
|
||||||
br.cs = br.cs[:0]
|
br.cs = br.cs[:0]
|
||||||
|
|
||||||
|
@ -88,6 +94,8 @@ func (br *blockResult) clone() *blockResult {
|
||||||
}
|
}
|
||||||
brNew.csBuf = csNew
|
brNew.csBuf = csNew
|
||||||
|
|
||||||
|
// do not clone br.csEmpty - it will be populated by the caller via getColumnByName().
|
||||||
|
|
||||||
return brNew
|
return brNew
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1325,8 +1333,21 @@ func (br *blockResult) getColumnByName(columnName string) *blockResultColumn {
|
||||||
return cs[idx]
|
return cs[idx]
|
||||||
}
|
}
|
||||||
|
|
||||||
br.addConstColumn(columnName, "")
|
// Search for empty column with the given name
|
||||||
return &br.csBuf[len(br.csBuf)-1]
|
csEmpty := br.csEmpty
|
||||||
|
for i := range csEmpty {
|
||||||
|
if csEmpty[i].name == columnName {
|
||||||
|
return &csEmpty[i]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create missing empty column
|
||||||
|
br.csEmpty = append(br.csEmpty, blockResultColumn{
|
||||||
|
name: br.a.copyString(columnName),
|
||||||
|
isConst: true,
|
||||||
|
valuesEncoded: getEmptyStrings(1),
|
||||||
|
})
|
||||||
|
return &br.csEmpty[len(br.csEmpty)-1]
|
||||||
}
|
}
|
||||||
|
|
||||||
func (br *blockResult) getColumns() []*blockResultColumn {
|
func (br *blockResult) getColumns() []*blockResultColumn {
|
||||||
|
|
|
@ -2,6 +2,9 @@ package logstorage
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"unsafe"
|
||||||
|
|
||||||
|
"github.com/VictoriaMetrics/VictoriaMetrics/lib/slicesutil"
|
||||||
)
|
)
|
||||||
|
|
||||||
// pipeExtract processes '| extract ...' pipe.
|
// pipeExtract processes '| extract ...' pipe.
|
||||||
|
@ -99,20 +102,121 @@ 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 {
|
||||||
patterns := make([]*pattern, workersCount)
|
return &pipeExtractProcessor{
|
||||||
for i := range patterns {
|
pe: pe,
|
||||||
patterns[i] = pe.ptn.clone()
|
ppBase: ppBase,
|
||||||
}
|
|
||||||
|
|
||||||
unpackFunc := func(uctx *fieldsUnpackerContext, s string) {
|
shards: make([]pipeExtractProcessorShard, workersCount),
|
||||||
ptn := patterns[uctx.workerID]
|
|
||||||
ptn.apply(s)
|
|
||||||
for _, f := range ptn.fields {
|
|
||||||
uctx.addField(f.name, *f.value)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return newPipeUnpackProcessor(workersCount, unpackFunc, ppBase, pe.fromField, "", pe.keepOriginalFields, pe.skipEmptyResults, pe.iff)
|
type pipeExtractProcessor struct {
|
||||||
|
pe *pipeExtract
|
||||||
|
ppBase pipeProcessor
|
||||||
|
|
||||||
|
shards []pipeExtractProcessorShard
|
||||||
|
}
|
||||||
|
|
||||||
|
type pipeExtractProcessorShard struct {
|
||||||
|
pipeExtractProcessorShardNopad
|
||||||
|
|
||||||
|
// The padding prevents false sharing on widespread platforms with 128 mod (cache line size) = 0 .
|
||||||
|
_ [128 - unsafe.Sizeof(pipeExtractProcessorShardNopad{})%128]byte
|
||||||
|
}
|
||||||
|
|
||||||
|
type pipeExtractProcessorShardNopad struct {
|
||||||
|
bm bitmap
|
||||||
|
ptn *pattern
|
||||||
|
|
||||||
|
resultValues []string
|
||||||
|
rcs []resultColumn
|
||||||
|
a arena
|
||||||
|
}
|
||||||
|
|
||||||
|
func (pep *pipeExtractProcessor) writeBlock(workerID uint, br *blockResult) {
|
||||||
|
if len(br.timestamps) == 0 {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
pe := pep.pe
|
||||||
|
shard := &pep.shards[workerID]
|
||||||
|
|
||||||
|
bm := &shard.bm
|
||||||
|
bm.init(len(br.timestamps))
|
||||||
|
bm.setBits()
|
||||||
|
if iff := pe.iff; iff != nil {
|
||||||
|
iff.f.applyToBlockResult(br, bm)
|
||||||
|
if bm.isZero() {
|
||||||
|
pep.ppBase.writeBlock(workerID, br)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if shard.ptn == nil {
|
||||||
|
shard.ptn = pe.ptn.clone()
|
||||||
|
}
|
||||||
|
ptn := shard.ptn
|
||||||
|
|
||||||
|
shard.rcs = slicesutil.SetLength(shard.rcs, len(ptn.fields))
|
||||||
|
rcs := shard.rcs
|
||||||
|
for i := range ptn.fields {
|
||||||
|
rcs[i].name = ptn.fields[i].name
|
||||||
|
}
|
||||||
|
|
||||||
|
c := br.getColumnByName(pe.fromField)
|
||||||
|
values := c.getValues(br)
|
||||||
|
|
||||||
|
shard.resultValues = slicesutil.SetLength(shard.resultValues, len(rcs))
|
||||||
|
resultValues := shard.resultValues
|
||||||
|
hadUpdates := false
|
||||||
|
vPrev := ""
|
||||||
|
for rowIdx, v := range values {
|
||||||
|
if bm.isSetBit(rowIdx) {
|
||||||
|
if !hadUpdates || vPrev != v {
|
||||||
|
vPrev = v
|
||||||
|
hadUpdates = true
|
||||||
|
|
||||||
|
ptn.apply(v)
|
||||||
|
|
||||||
|
for i, f := range ptn.fields {
|
||||||
|
v := *f.value
|
||||||
|
if v == "" && pe.skipEmptyResults || pe.keepOriginalFields {
|
||||||
|
c := br.getColumnByName(rcs[i].name)
|
||||||
|
if vOrig := c.getValueAtRow(br, rowIdx); vOrig != "" {
|
||||||
|
v = vOrig
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
v = shard.a.copyString(v)
|
||||||
|
}
|
||||||
|
resultValues[i] = v
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
for i := range rcs {
|
||||||
|
c := br.getColumnByName(rcs[i].name)
|
||||||
|
v := c.getValueAtRow(br, rowIdx)
|
||||||
|
resultValues[i] = v
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for i, v := range resultValues {
|
||||||
|
rcs[i].addValue(v)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for i := range rcs {
|
||||||
|
br.addResultColumn(&rcs[i])
|
||||||
|
}
|
||||||
|
pep.ppBase.writeBlock(workerID, br)
|
||||||
|
|
||||||
|
for i := range rcs {
|
||||||
|
rcs[i].reset()
|
||||||
|
}
|
||||||
|
shard.a.reset()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (pep *pipeExtractProcessor) flush() error {
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func parsePipeExtract(lex *lexer) (*pipeExtract, error) {
|
func parsePipeExtract(lex *lexer) (*pipeExtract, error) {
|
||||||
|
|
|
@ -166,7 +166,6 @@ func TestPipeUnpackJSON(t *testing.T) {
|
||||||
}, [][]Field{
|
}, [][]Field{
|
||||||
{
|
{
|
||||||
{"_msg", `{"foo":"bar"}`},
|
{"_msg", `{"foo":"bar"}`},
|
||||||
{"x", ""},
|
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
|
||||||
|
@ -313,7 +312,6 @@ func TestPipeUnpackJSON(t *testing.T) {
|
||||||
{"y", `abc`},
|
{"y", `abc`},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
{"y", ""},
|
|
||||||
{"z", `foobar`},
|
{"z", `foobar`},
|
||||||
{"x", `{"z":["bar",123]}`},
|
{"x", `{"z":["bar",123]}`},
|
||||||
},
|
},
|
||||||
|
|
|
@ -151,7 +151,6 @@ func TestPipeUnpackLogfmt(t *testing.T) {
|
||||||
},
|
},
|
||||||
}, [][]Field{
|
}, [][]Field{
|
||||||
{
|
{
|
||||||
{"foo", ""},
|
|
||||||
{"_msg", `foo=bar baz="x y=z" a=b`},
|
{"_msg", `foo=bar baz="x y=z" a=b`},
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
@ -291,7 +290,6 @@ func TestPipeUnpackLogfmt(t *testing.T) {
|
||||||
{"y", `abc`},
|
{"y", `abc`},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
{"y", ""},
|
|
||||||
{"z", `foobar`},
|
{"z", `foobar`},
|
||||||
{"x", `z=bar`},
|
{"x", `z=bar`},
|
||||||
},
|
},
|
||||||
|
|
|
@ -275,7 +275,7 @@ func TestStatsFieldsMax(t *testing.T) {
|
||||||
{
|
{
|
||||||
{"a", "1"},
|
{"a", "1"},
|
||||||
{"b", ""},
|
{"b", ""},
|
||||||
{"x", `{"_msg":"def","a":"1","c":"foo","b":""}`},
|
{"x", `{"_msg":"def","a":"1","c":"foo"}`},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
{"a", "3"},
|
{"a", "3"},
|
||||||
|
|
|
@ -274,7 +274,7 @@ func TestStatsFieldsMin(t *testing.T) {
|
||||||
{
|
{
|
||||||
{"a", "1"},
|
{"a", "1"},
|
||||||
{"b", ""},
|
{"b", ""},
|
||||||
{"x", `{"_msg":"def","a":"1","c":"foo","b":""}`},
|
{"x", `{"_msg":"def","a":"1","c":"foo"}`},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
{"a", "3"},
|
{"a", "3"},
|
||||||
|
|
Loading…
Reference in a new issue