2024-05-03 10:54:37 +00:00
|
|
|
package logstorage
|
|
|
|
|
|
|
|
import (
|
2024-05-03 11:44:57 +00:00
|
|
|
"slices"
|
2024-05-03 10:54:37 +00:00
|
|
|
"sort"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
"unsafe"
|
|
|
|
|
|
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/lib/bytesutil"
|
|
|
|
)
|
|
|
|
|
|
|
|
type statsUniqArray struct {
|
2024-05-03 11:44:57 +00:00
|
|
|
fields []string
|
|
|
|
containsStar bool
|
2024-05-03 10:54:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (su *statsUniqArray) String() string {
|
2024-05-03 11:44:57 +00:00
|
|
|
return "uniq_array(" + fieldNamesString(su.fields) + ")"
|
2024-05-03 10:54:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (su *statsUniqArray) neededFields() []string {
|
2024-05-03 11:44:57 +00:00
|
|
|
return su.fields
|
2024-05-03 10:54:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (su *statsUniqArray) newStatsProcessor() (statsProcessor, int) {
|
|
|
|
sup := &statsUniqArrayProcessor{
|
|
|
|
su: su,
|
|
|
|
|
|
|
|
m: make(map[string]struct{}),
|
|
|
|
}
|
|
|
|
return sup, int(unsafe.Sizeof(*sup))
|
|
|
|
}
|
|
|
|
|
|
|
|
type statsUniqArrayProcessor struct {
|
|
|
|
su *statsUniqArray
|
|
|
|
|
|
|
|
m map[string]struct{}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (sup *statsUniqArrayProcessor) updateStatsForAllRows(br *blockResult) int {
|
2024-05-03 11:44:57 +00:00
|
|
|
stateSizeIncrease := 0
|
2024-05-03 12:03:17 +00:00
|
|
|
if sup.su.containsStar {
|
2024-05-03 11:44:57 +00:00
|
|
|
columns := br.getColumns()
|
|
|
|
for i := range columns {
|
|
|
|
stateSizeIncrease += sup.updateStatsForAllRowsColumn(&columns[i], br)
|
|
|
|
}
|
|
|
|
} else {
|
2024-05-03 12:03:17 +00:00
|
|
|
for _, field := range sup.su.fields {
|
2024-05-03 11:44:57 +00:00
|
|
|
c := br.getColumnByName(field)
|
|
|
|
stateSizeIncrease += sup.updateStatsForAllRowsColumn(&c, br)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return stateSizeIncrease
|
|
|
|
}
|
2024-05-03 10:54:37 +00:00
|
|
|
|
2024-05-03 11:44:57 +00:00
|
|
|
func (sup *statsUniqArrayProcessor) updateStatsForAllRowsColumn(c *blockResultColumn, br *blockResult) int {
|
|
|
|
m := sup.m
|
2024-05-03 10:54:37 +00:00
|
|
|
stateSizeIncrease := 0
|
|
|
|
if c.isConst {
|
|
|
|
// collect unique const values
|
|
|
|
v := c.encodedValues[0]
|
|
|
|
if v == "" {
|
|
|
|
// skip empty values
|
|
|
|
return stateSizeIncrease
|
|
|
|
}
|
|
|
|
if _, ok := m[v]; !ok {
|
|
|
|
vCopy := strings.Clone(v)
|
|
|
|
m[vCopy] = struct{}{}
|
|
|
|
stateSizeIncrease += len(vCopy) + int(unsafe.Sizeof(vCopy))
|
|
|
|
}
|
|
|
|
return stateSizeIncrease
|
|
|
|
}
|
|
|
|
if c.valueType == valueTypeDict {
|
|
|
|
// collect unique non-zero c.dictValues
|
|
|
|
for _, v := range c.dictValues {
|
|
|
|
if v == "" {
|
|
|
|
// skip empty values
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if _, ok := m[v]; !ok {
|
|
|
|
vCopy := strings.Clone(v)
|
|
|
|
m[vCopy] = struct{}{}
|
|
|
|
stateSizeIncrease += len(vCopy) + int(unsafe.Sizeof(vCopy))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return stateSizeIncrease
|
|
|
|
}
|
|
|
|
|
|
|
|
// slow path - collect unique values across all rows
|
|
|
|
values := c.getValues(br)
|
|
|
|
for i, v := range values {
|
|
|
|
if v == "" {
|
|
|
|
// skip empty values
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if i > 0 && values[i-1] == v {
|
|
|
|
// This value has been already counted.
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if _, ok := m[v]; !ok {
|
|
|
|
vCopy := strings.Clone(v)
|
|
|
|
m[vCopy] = struct{}{}
|
|
|
|
stateSizeIncrease += len(vCopy) + int(unsafe.Sizeof(vCopy))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return stateSizeIncrease
|
|
|
|
}
|
|
|
|
|
|
|
|
func (sup *statsUniqArrayProcessor) updateStatsForRow(br *blockResult, rowIdx int) int {
|
|
|
|
stateSizeIncrease := 0
|
2024-05-03 12:03:17 +00:00
|
|
|
if sup.su.containsStar {
|
2024-05-03 11:44:57 +00:00
|
|
|
columns := br.getColumns()
|
|
|
|
for i := range columns {
|
|
|
|
stateSizeIncrease += sup.updateStatsForRowColumn(&columns[i], br, rowIdx)
|
|
|
|
}
|
|
|
|
} else {
|
2024-05-03 12:03:17 +00:00
|
|
|
for _, field := range sup.su.fields {
|
2024-05-03 11:44:57 +00:00
|
|
|
c := br.getColumnByName(field)
|
|
|
|
stateSizeIncrease += sup.updateStatsForRowColumn(&c, br, rowIdx)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return stateSizeIncrease
|
|
|
|
}
|
|
|
|
|
|
|
|
func (sup *statsUniqArrayProcessor) updateStatsForRowColumn(c *blockResultColumn, br *blockResult, rowIdx int) int {
|
|
|
|
m := sup.m
|
|
|
|
stateSizeIncrease := 0
|
2024-05-03 10:54:37 +00:00
|
|
|
if c.isConst {
|
|
|
|
// collect unique const values
|
|
|
|
v := c.encodedValues[0]
|
|
|
|
if v == "" {
|
|
|
|
// skip empty values
|
|
|
|
return stateSizeIncrease
|
|
|
|
}
|
|
|
|
if _, ok := m[v]; !ok {
|
|
|
|
vCopy := strings.Clone(v)
|
|
|
|
m[vCopy] = struct{}{}
|
|
|
|
stateSizeIncrease += len(vCopy) + int(unsafe.Sizeof(vCopy))
|
|
|
|
}
|
|
|
|
return stateSizeIncrease
|
|
|
|
}
|
|
|
|
if c.valueType == valueTypeDict {
|
|
|
|
// collect unique non-zero c.dictValues
|
|
|
|
dictIdx := c.encodedValues[rowIdx][0]
|
|
|
|
v := c.dictValues[dictIdx]
|
|
|
|
if v == "" {
|
|
|
|
// skip empty values
|
|
|
|
return stateSizeIncrease
|
|
|
|
}
|
|
|
|
if _, ok := m[v]; !ok {
|
|
|
|
vCopy := strings.Clone(v)
|
|
|
|
m[vCopy] = struct{}{}
|
|
|
|
stateSizeIncrease += len(vCopy) + int(unsafe.Sizeof(vCopy))
|
|
|
|
}
|
|
|
|
return stateSizeIncrease
|
|
|
|
}
|
|
|
|
|
|
|
|
// collect unique values for the given rowIdx.
|
|
|
|
v := c.getValueAtRow(br, rowIdx)
|
|
|
|
if v == "" {
|
|
|
|
// skip empty values
|
|
|
|
return stateSizeIncrease
|
|
|
|
}
|
|
|
|
if _, ok := m[v]; !ok {
|
|
|
|
vCopy := strings.Clone(v)
|
|
|
|
m[vCopy] = struct{}{}
|
|
|
|
stateSizeIncrease += len(vCopy) + int(unsafe.Sizeof(vCopy))
|
|
|
|
}
|
|
|
|
return stateSizeIncrease
|
|
|
|
}
|
|
|
|
|
|
|
|
func (sup *statsUniqArrayProcessor) mergeState(sfp statsProcessor) {
|
|
|
|
src := sfp.(*statsUniqArrayProcessor)
|
|
|
|
m := sup.m
|
|
|
|
for k := range src.m {
|
|
|
|
if _, ok := m[k]; !ok {
|
|
|
|
m[k] = struct{}{}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (sup *statsUniqArrayProcessor) finalizeStats() string {
|
|
|
|
if len(sup.m) == 0 {
|
|
|
|
return "[]"
|
|
|
|
}
|
|
|
|
|
|
|
|
// Sort unique items
|
|
|
|
items := make([]string, 0, len(sup.m))
|
|
|
|
for k := range sup.m {
|
|
|
|
items = append(items, k)
|
|
|
|
}
|
|
|
|
sort.Strings(items)
|
|
|
|
|
|
|
|
// Marshal items into JSON array.
|
|
|
|
|
|
|
|
// Pre-allocate buffer for serialized items.
|
|
|
|
// Assume that there is no need in quoting items. Otherwise additional reallocations
|
|
|
|
// for the allocated buffer are possible.
|
|
|
|
bufSize := len(items) + 1
|
|
|
|
for _, item := range items {
|
|
|
|
bufSize += len(item)
|
|
|
|
}
|
|
|
|
b := make([]byte, 0, bufSize)
|
|
|
|
|
|
|
|
b = append(b, '[')
|
|
|
|
b = strconv.AppendQuote(b, items[0])
|
|
|
|
for _, item := range items[1:] {
|
|
|
|
b = append(b, ',')
|
|
|
|
b = strconv.AppendQuote(b, item)
|
|
|
|
}
|
|
|
|
b = append(b, ']')
|
|
|
|
|
|
|
|
return bytesutil.ToUnsafeString(b)
|
|
|
|
}
|
|
|
|
|
|
|
|
func parseStatsUniqArray(lex *lexer) (*statsUniqArray, error) {
|
2024-05-03 12:03:17 +00:00
|
|
|
fields, err := parseFieldNamesForStatsFunc(lex, "uniq_array")
|
2024-05-03 10:54:37 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
su := &statsUniqArray{
|
2024-05-03 11:44:57 +00:00
|
|
|
fields: fields,
|
|
|
|
containsStar: slices.Contains(fields, "*"),
|
2024-05-03 10:54:37 +00:00
|
|
|
}
|
|
|
|
return su, nil
|
|
|
|
}
|