mirror of
https://github.com/VictoriaMetrics/VictoriaMetrics.git
synced 2025-01-20 07:19:17 +01:00
lib/logstorage: return the expected hits
results from uniq
pipe when the number of unique values reaches the specified limit
Previously `uniq` pipe could return zero `hits` if the number of found unique values equals the specified limit. This wasn't expected in most cases.
This commit is contained in:
parent
58d1e517de
commit
fce6fc745a
@ -135,7 +135,7 @@ type pipeUniqProcessorShardNopad struct {
|
||||
//
|
||||
// It returns false if the block cannot be written because of the exceeded limit.
|
||||
func (shard *pipeUniqProcessorShard) writeBlock(br *blockResult) bool {
|
||||
if limit := shard.pu.limit; limit > 0 && uint64(len(shard.m)) >= limit {
|
||||
if limit := shard.pu.limit; limit > 0 && uint64(len(shard.m)) > limit {
|
||||
return false
|
||||
}
|
||||
|
||||
@ -301,7 +301,7 @@ func (pup *pipeUniqProcessor) flush() error {
|
||||
|
||||
// There is little sense in returning partial hits when the limit on the number of unique entries is reached.
|
||||
// It is better from UX experience is to return zero hits instead.
|
||||
resetHits := pup.pu.limit > 0 && uint64(len(m)) >= pup.pu.limit
|
||||
resetHits := pup.pu.limit > 0 && uint64(len(m)) > pup.pu.limit
|
||||
|
||||
// write result
|
||||
wctx := &pipeUniqWriteContext{
|
||||
|
@ -114,13 +114,41 @@ func TestPipeUniq(t *testing.T) {
|
||||
{
|
||||
{"a", "2"},
|
||||
{"b", "3"},
|
||||
{"hits", "0"},
|
||||
{"hits", "2"},
|
||||
},
|
||||
{
|
||||
{"a", `2`},
|
||||
{"b", `54`},
|
||||
{"c", "d"},
|
||||
{"hits", "0"},
|
||||
{"hits", "1"},
|
||||
},
|
||||
})
|
||||
|
||||
f("uniq hits limit 3", [][]Field{
|
||||
{
|
||||
{"a", `2`},
|
||||
{"b", `3`},
|
||||
},
|
||||
{
|
||||
{"a", "2"},
|
||||
{"b", "3"},
|
||||
},
|
||||
{
|
||||
{"a", `2`},
|
||||
{"b", `54`},
|
||||
{"c", "d"},
|
||||
},
|
||||
}, [][]Field{
|
||||
{
|
||||
{"a", "2"},
|
||||
{"b", "3"},
|
||||
{"hits", "2"},
|
||||
},
|
||||
{
|
||||
{"a", `2`},
|
||||
{"b", `54`},
|
||||
{"c", "d"},
|
||||
{"hits", "1"},
|
||||
},
|
||||
})
|
||||
|
||||
|
@ -355,7 +355,7 @@ func (sup *statsCountUniqProcessor) updateState(v []byte) int {
|
||||
|
||||
func (sup *statsCountUniqProcessor) limitReached() bool {
|
||||
limit := sup.su.limit
|
||||
return limit > 0 && uint64(len(sup.m)) >= limit
|
||||
return limit > 0 && uint64(len(sup.m)) > limit
|
||||
}
|
||||
|
||||
func parseStatsCountUniq(lex *lexer) (*statsCountUniq, error) {
|
||||
|
@ -215,7 +215,7 @@ func (sup *statsUniqValuesProcessor) updateState(v string) int {
|
||||
|
||||
func (sup *statsUniqValuesProcessor) limitReached() bool {
|
||||
limit := sup.su.limit
|
||||
return limit > 0 && uint64(len(sup.m)) >= limit
|
||||
return limit > 0 && uint64(len(sup.m)) > limit
|
||||
}
|
||||
|
||||
func marshalJSONArray(items []string) string {
|
||||
|
@ -184,7 +184,7 @@ func (svp *statsValuesProcessor) finalizeStats() string {
|
||||
|
||||
func (svp *statsValuesProcessor) limitReached() bool {
|
||||
limit := svp.sv.limit
|
||||
return limit > 0 && uint64(len(svp.values)) >= limit
|
||||
return limit > 0 && uint64(len(svp.values)) > limit
|
||||
}
|
||||
|
||||
func parseStatsValues(lex *lexer) (*statsValues, error) {
|
||||
|
@ -384,9 +384,9 @@ func TestStorageRunQuery(t *testing.T) {
|
||||
}
|
||||
|
||||
resultsExpected := []ValueWithHits{
|
||||
{`{instance="host-0:234",job="foobar"}`, 0},
|
||||
{`{instance="host-1:234",job="foobar"}`, 0},
|
||||
{`{instance="host-2:234",job="foobar"}`, 0},
|
||||
{`{instance="host-0:234",job="foobar"}`, 385},
|
||||
{`{instance="host-1:234",job="foobar"}`, 385},
|
||||
{`{instance="host-2:234",job="foobar"}`, 385},
|
||||
}
|
||||
if !reflect.DeepEqual(results, resultsExpected) {
|
||||
t.Fatalf("unexpected result; got\n%v\nwant\n%v", results, resultsExpected)
|
||||
|
Loading…
Reference in New Issue
Block a user