VictoriaMetrics/lib/logstorage/stats_row_min.go

252 lines
5.8 KiB
Go
Raw Normal View History

2024-05-22 19:01:20 +00:00
package logstorage
import (
"fmt"
"math"
"slices"
"strings"
"unsafe"
"github.com/VictoriaMetrics/VictoriaMetrics/lib/bytesutil"
"github.com/VictoriaMetrics/VictoriaMetrics/lib/logger"
)
2024-05-30 14:19:23 +00:00
type statsRowMin struct {
2024-05-22 19:01:20 +00:00
srcField string
fetchFields []string
}
2024-05-30 14:19:23 +00:00
func (sm *statsRowMin) String() string {
s := "row_min(" + quoteTokenIfNeeded(sm.srcField)
2024-05-22 19:01:20 +00:00
if len(sm.fetchFields) > 0 {
s += ", " + fieldNamesString(sm.fetchFields)
}
s += ")"
return s
}
2024-05-30 14:19:23 +00:00
func (sm *statsRowMin) updateNeededFields(neededFields fieldsSet) {
2024-05-22 19:01:20 +00:00
if len(sm.fetchFields) == 0 {
neededFields.add("*")
} else {
neededFields.addFields(sm.fetchFields)
}
neededFields.add(sm.srcField)
}
2024-05-30 14:19:23 +00:00
func (sm *statsRowMin) newStatsProcessor() (statsProcessor, int) {
smp := &statsRowMinProcessor{
2024-05-22 19:01:20 +00:00
sm: sm,
}
return smp, int(unsafe.Sizeof(*smp))
}
2024-05-30 14:19:23 +00:00
type statsRowMinProcessor struct {
sm *statsRowMin
2024-05-22 19:01:20 +00:00
min string
fields []Field
}
2024-05-30 14:19:23 +00:00
func (smp *statsRowMinProcessor) updateStatsForAllRows(br *blockResult) int {
2024-05-22 19:01:20 +00:00
stateSizeIncrease := 0
c := br.getColumnByName(smp.sm.srcField)
if c.isConst {
v := c.valuesEncoded[0]
stateSizeIncrease += smp.updateState(v, br, 0)
return stateSizeIncrease
}
if c.isTime {
timestamp, ok := TryParseTimestampRFC3339Nano(smp.min)
if !ok {
timestamp = (1 << 63) - 1
}
minTimestamp := br.getMinTimestamp(timestamp)
if minTimestamp >= timestamp {
return stateSizeIncrease
}
2024-05-22 19:01:20 +00:00
bb := bbPool.Get()
bb.B = marshalTimestampRFC3339NanoString(bb.B[:0], minTimestamp)
2024-05-22 19:01:20 +00:00
v := bytesutil.ToUnsafeString(bb.B)
stateSizeIncrease += smp.updateState(v, br, 0)
bbPool.Put(bb)
return stateSizeIncrease
}
needUpdateState := false
switch c.valueType {
case valueTypeString:
needUpdateState = true
case valueTypeDict:
for _, v := range c.dictValues {
if smp.needUpdateStateString(v) {
needUpdateState = true
break
}
}
case valueTypeUint8, valueTypeUint16, valueTypeUint32, valueTypeUint64:
bb := bbPool.Get()
bb.B = marshalUint64String(bb.B[:0], c.minValue)
needUpdateState = smp.needUpdateStateBytes(bb.B)
bbPool.Put(bb)
case valueTypeFloat64:
f := math.Float64frombits(c.minValue)
bb := bbPool.Get()
bb.B = marshalFloat64String(bb.B[:0], f)
needUpdateState = smp.needUpdateStateBytes(bb.B)
bbPool.Put(bb)
case valueTypeIPv4:
bb := bbPool.Get()
bb.B = marshalIPv4String(bb.B[:0], uint32(c.minValue))
needUpdateState = smp.needUpdateStateBytes(bb.B)
bbPool.Put(bb)
case valueTypeTimestampISO8601:
bb := bbPool.Get()
bb.B = marshalTimestampISO8601String(bb.B[:0], int64(c.minValue))
needUpdateState = smp.needUpdateStateBytes(bb.B)
bbPool.Put(bb)
default:
logger.Panicf("BUG: unknown valueType=%d", c.valueType)
}
if needUpdateState {
values := c.getValues(br)
for i, v := range values {
stateSizeIncrease += smp.updateState(v, br, i)
}
}
return stateSizeIncrease
}
2024-05-30 14:19:23 +00:00
func (smp *statsRowMinProcessor) updateStatsForRow(br *blockResult, rowIdx int) int {
2024-05-22 19:01:20 +00:00
stateSizeIncrease := 0
c := br.getColumnByName(smp.sm.srcField)
if c.isConst {
v := c.valuesEncoded[0]
stateSizeIncrease += smp.updateState(v, br, rowIdx)
return stateSizeIncrease
}
if c.isTime {
timestamps := br.getTimestamps()
2024-05-22 19:01:20 +00:00
bb := bbPool.Get()
bb.B = marshalTimestampRFC3339NanoString(bb.B[:0], timestamps[rowIdx])
2024-05-22 19:01:20 +00:00
v := bytesutil.ToUnsafeString(bb.B)
stateSizeIncrease += smp.updateState(v, br, rowIdx)
bbPool.Put(bb)
return stateSizeIncrease
}
v := c.getValueAtRow(br, rowIdx)
stateSizeIncrease += smp.updateState(v, br, rowIdx)
return stateSizeIncrease
}
2024-05-30 14:19:23 +00:00
func (smp *statsRowMinProcessor) mergeState(sfp statsProcessor) {
src := sfp.(*statsRowMinProcessor)
2024-05-22 19:01:20 +00:00
if smp.needUpdateStateString(src.min) {
smp.min = src.min
smp.fields = src.fields
}
}
2024-05-30 14:19:23 +00:00
func (smp *statsRowMinProcessor) needUpdateStateBytes(b []byte) bool {
2024-05-22 19:01:20 +00:00
v := bytesutil.ToUnsafeString(b)
return smp.needUpdateStateString(v)
}
2024-05-30 14:19:23 +00:00
func (smp *statsRowMinProcessor) needUpdateStateString(v string) bool {
2024-05-22 19:01:20 +00:00
if v == "" {
return false
}
return smp.min == "" || lessString(v, smp.min)
}
2024-05-30 14:19:23 +00:00
func (smp *statsRowMinProcessor) updateState(v string, br *blockResult, rowIdx int) int {
2024-05-22 19:01:20 +00:00
stateSizeIncrease := 0
if !smp.needUpdateStateString(v) {
// There is no need in updating state
return stateSizeIncrease
}
stateSizeIncrease -= len(smp.min)
stateSizeIncrease += len(v)
smp.min = strings.Clone(v)
fields := smp.fields
for _, f := range fields {
stateSizeIncrease -= len(f.Name) + len(f.Value)
}
clear(fields)
fields = fields[:0]
fetchFields := smp.sm.fetchFields
if len(fetchFields) == 0 {
cs := br.getColumns()
for _, c := range cs {
v := c.getValueAtRow(br, rowIdx)
fields = append(fields, Field{
Name: strings.Clone(c.name),
Value: strings.Clone(v),
})
stateSizeIncrease += len(c.name) + len(v)
}
} else {
for _, field := range fetchFields {
c := br.getColumnByName(field)
v := c.getValueAtRow(br, rowIdx)
fields = append(fields, Field{
Name: strings.Clone(c.name),
Value: strings.Clone(v),
})
stateSizeIncrease += len(c.name) + len(v)
}
}
smp.fields = fields
return stateSizeIncrease
}
2024-05-30 14:19:23 +00:00
func (smp *statsRowMinProcessor) finalizeStats() string {
2024-05-22 19:01:20 +00:00
bb := bbPool.Get()
2024-06-03 22:59:25 +00:00
bb.B = MarshalFieldsToJSON(bb.B, smp.fields)
2024-05-22 19:01:20 +00:00
result := string(bb.B)
bbPool.Put(bb)
return result
}
2024-05-30 14:19:23 +00:00
func parseStatsRowMin(lex *lexer) (*statsRowMin, error) {
if !lex.isKeyword("row_min") {
return nil, fmt.Errorf("unexpected func; got %q; want 'row_min'", lex.token)
2024-05-22 19:01:20 +00:00
}
lex.nextToken()
fields, err := parseFieldNamesInParens(lex)
if err != nil {
2024-05-30 14:19:23 +00:00
return nil, fmt.Errorf("cannot parse 'row_min' args: %w", err)
2024-05-22 19:01:20 +00:00
}
if len(fields) == 0 {
2024-05-30 14:19:23 +00:00
return nil, fmt.Errorf("missing first arg for 'row_min' func - source field")
2024-05-22 19:01:20 +00:00
}
srcField := fields[0]
fetchFields := fields[1:]
if slices.Contains(fetchFields, "*") {
fetchFields = nil
}
2024-05-30 14:19:23 +00:00
sm := &statsRowMin{
2024-05-22 19:01:20 +00:00
srcField: srcField,
fetchFields: fetchFields,
}
return sm, nil
}