582 lines
18 KiB
Go
582 lines
18 KiB
Go
package query_test
|
|
|
|
import (
|
|
"crypto/sha1"
|
|
"fmt"
|
|
"math"
|
|
"math/rand"
|
|
"strconv"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/davecgh/go-spew/spew"
|
|
"github.com/influxdata/influxdb/pkg/deep"
|
|
"github.com/influxdata/influxdb/query"
|
|
"github.com/influxdata/influxql"
|
|
tassert "github.com/stretchr/testify/assert"
|
|
trequire "github.com/stretchr/testify/require"
|
|
)
|
|
|
|
func almostEqual(got, exp float64) bool {
|
|
return math.Abs(got-exp) < 1e-5 && !math.IsNaN(got)
|
|
}
|
|
|
|
func TestHoltWinters_AusTourists(t *testing.T) {
|
|
hw := query.NewFloatHoltWintersReducer(10, 4, false, 1)
|
|
// Dataset from http://www.inside-r.org/packages/cran/fpp/docs/austourists
|
|
austourists := []query.FloatPoint{
|
|
{Time: 1, Value: 30.052513},
|
|
{Time: 2, Value: 19.148496},
|
|
{Time: 3, Value: 25.317692},
|
|
{Time: 4, Value: 27.591437},
|
|
{Time: 5, Value: 32.076456},
|
|
{Time: 6, Value: 23.487961},
|
|
{Time: 7, Value: 28.47594},
|
|
{Time: 8, Value: 35.123753},
|
|
{Time: 9, Value: 36.838485},
|
|
{Time: 10, Value: 25.007017},
|
|
{Time: 11, Value: 30.72223},
|
|
{Time: 12, Value: 28.693759},
|
|
{Time: 13, Value: 36.640986},
|
|
{Time: 14, Value: 23.824609},
|
|
{Time: 15, Value: 29.311683},
|
|
{Time: 16, Value: 31.770309},
|
|
{Time: 17, Value: 35.177877},
|
|
{Time: 18, Value: 19.775244},
|
|
{Time: 19, Value: 29.60175},
|
|
{Time: 20, Value: 34.538842},
|
|
{Time: 21, Value: 41.273599},
|
|
{Time: 22, Value: 26.655862},
|
|
{Time: 23, Value: 28.279859},
|
|
{Time: 24, Value: 35.191153},
|
|
{Time: 25, Value: 41.727458},
|
|
{Time: 26, Value: 24.04185},
|
|
{Time: 27, Value: 32.328103},
|
|
{Time: 28, Value: 37.328708},
|
|
{Time: 29, Value: 46.213153},
|
|
{Time: 30, Value: 29.346326},
|
|
{Time: 31, Value: 36.48291},
|
|
{Time: 32, Value: 42.977719},
|
|
{Time: 33, Value: 48.901525},
|
|
{Time: 34, Value: 31.180221},
|
|
{Time: 35, Value: 37.717881},
|
|
{Time: 36, Value: 40.420211},
|
|
{Time: 37, Value: 51.206863},
|
|
{Time: 38, Value: 31.887228},
|
|
{Time: 39, Value: 40.978263},
|
|
{Time: 40, Value: 43.772491},
|
|
{Time: 41, Value: 55.558567},
|
|
{Time: 42, Value: 33.850915},
|
|
{Time: 43, Value: 42.076383},
|
|
{Time: 44, Value: 45.642292},
|
|
{Time: 45, Value: 59.76678},
|
|
{Time: 46, Value: 35.191877},
|
|
{Time: 47, Value: 44.319737},
|
|
{Time: 48, Value: 47.913736},
|
|
}
|
|
|
|
for _, p := range austourists {
|
|
hw.AggregateFloat(&p)
|
|
}
|
|
points := hw.Emit()
|
|
|
|
forecasted := []query.FloatPoint{
|
|
{Time: 49, Value: 51.85064132137853},
|
|
{Time: 50, Value: 43.26055282315273},
|
|
{Time: 51, Value: 41.827258044814464},
|
|
{Time: 52, Value: 54.3990354591749},
|
|
{Time: 53, Value: 54.62334472770803},
|
|
{Time: 54, Value: 45.57155693625209},
|
|
{Time: 55, Value: 44.06051240252263},
|
|
{Time: 56, Value: 57.30029870759433},
|
|
{Time: 57, Value: 57.53591513519172},
|
|
{Time: 58, Value: 47.999008139396096},
|
|
}
|
|
|
|
if exp, got := len(forecasted), len(points); exp != got {
|
|
t.Fatalf("unexpected number of points emitted: got %d exp %d", got, exp)
|
|
}
|
|
|
|
for i := range forecasted {
|
|
if exp, got := forecasted[i].Time, points[i].Time; got != exp {
|
|
t.Errorf("unexpected time on points[%d] got %v exp %v", i, got, exp)
|
|
}
|
|
if exp, got := forecasted[i].Value, points[i].Value; !almostEqual(got, exp) {
|
|
t.Errorf("unexpected value on points[%d] got %v exp %v", i, got, exp)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestHoltWinters_AusTourists_Missing(t *testing.T) {
|
|
hw := query.NewFloatHoltWintersReducer(10, 4, false, 1)
|
|
// Dataset from http://www.inside-r.org/packages/cran/fpp/docs/austourists
|
|
austourists := []query.FloatPoint{
|
|
{Time: 1, Value: 30.052513},
|
|
{Time: 3, Value: 25.317692},
|
|
{Time: 4, Value: 27.591437},
|
|
{Time: 5, Value: 32.076456},
|
|
{Time: 6, Value: 23.487961},
|
|
{Time: 7, Value: 28.47594},
|
|
{Time: 9, Value: 36.838485},
|
|
{Time: 10, Value: 25.007017},
|
|
{Time: 11, Value: 30.72223},
|
|
{Time: 12, Value: 28.693759},
|
|
{Time: 13, Value: 36.640986},
|
|
{Time: 14, Value: 23.824609},
|
|
{Time: 15, Value: 29.311683},
|
|
{Time: 16, Value: 31.770309},
|
|
{Time: 17, Value: 35.177877},
|
|
{Time: 19, Value: 29.60175},
|
|
{Time: 20, Value: 34.538842},
|
|
{Time: 21, Value: 41.273599},
|
|
{Time: 22, Value: 26.655862},
|
|
{Time: 23, Value: 28.279859},
|
|
{Time: 24, Value: 35.191153},
|
|
{Time: 25, Value: 41.727458},
|
|
{Time: 26, Value: 24.04185},
|
|
{Time: 27, Value: 32.328103},
|
|
{Time: 28, Value: 37.328708},
|
|
{Time: 30, Value: 29.346326},
|
|
{Time: 31, Value: 36.48291},
|
|
{Time: 32, Value: 42.977719},
|
|
{Time: 34, Value: 31.180221},
|
|
{Time: 35, Value: 37.717881},
|
|
{Time: 36, Value: 40.420211},
|
|
{Time: 37, Value: 51.206863},
|
|
{Time: 38, Value: 31.887228},
|
|
{Time: 41, Value: 55.558567},
|
|
{Time: 42, Value: 33.850915},
|
|
{Time: 43, Value: 42.076383},
|
|
{Time: 44, Value: 45.642292},
|
|
{Time: 45, Value: 59.76678},
|
|
{Time: 46, Value: 35.191877},
|
|
{Time: 47, Value: 44.319737},
|
|
{Time: 48, Value: 47.913736},
|
|
}
|
|
|
|
for _, p := range austourists {
|
|
hw.AggregateFloat(&p)
|
|
}
|
|
points := hw.Emit()
|
|
|
|
forecasted := []query.FloatPoint{
|
|
{Time: 49, Value: 54.84533610387743},
|
|
{Time: 50, Value: 41.19329421863249},
|
|
{Time: 51, Value: 45.71673175112451},
|
|
{Time: 52, Value: 56.05759298805955},
|
|
{Time: 53, Value: 59.32337460282217},
|
|
{Time: 54, Value: 44.75280096850461},
|
|
{Time: 55, Value: 49.98865098113751},
|
|
{Time: 56, Value: 61.86084934967605},
|
|
{Time: 57, Value: 65.95805633454883},
|
|
{Time: 58, Value: 50.1502170480547},
|
|
}
|
|
|
|
if exp, got := len(forecasted), len(points); exp != got {
|
|
t.Fatalf("unexpected number of points emitted: got %d exp %d", got, exp)
|
|
}
|
|
|
|
for i := range forecasted {
|
|
if exp, got := forecasted[i].Time, points[i].Time; got != exp {
|
|
t.Errorf("unexpected time on points[%d] got %v exp %v", i, got, exp)
|
|
}
|
|
if exp, got := forecasted[i].Value, points[i].Value; !almostEqual(got, exp) {
|
|
t.Errorf("unexpected value on points[%d] got %v exp %v", i, got, exp)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestHoltWinters_USPopulation(t *testing.T) {
|
|
series := []query.FloatPoint{
|
|
{Time: 1, Value: 3.93},
|
|
{Time: 2, Value: 5.31},
|
|
{Time: 3, Value: 7.24},
|
|
{Time: 4, Value: 9.64},
|
|
{Time: 5, Value: 12.90},
|
|
{Time: 6, Value: 17.10},
|
|
{Time: 7, Value: 23.20},
|
|
{Time: 8, Value: 31.40},
|
|
{Time: 9, Value: 39.80},
|
|
{Time: 10, Value: 50.20},
|
|
{Time: 11, Value: 62.90},
|
|
{Time: 12, Value: 76.00},
|
|
{Time: 13, Value: 92.00},
|
|
{Time: 14, Value: 105.70},
|
|
{Time: 15, Value: 122.80},
|
|
{Time: 16, Value: 131.70},
|
|
{Time: 17, Value: 151.30},
|
|
{Time: 18, Value: 179.30},
|
|
{Time: 19, Value: 203.20},
|
|
}
|
|
hw := query.NewFloatHoltWintersReducer(10, 0, true, 1)
|
|
for _, p := range series {
|
|
hw.AggregateFloat(&p)
|
|
}
|
|
points := hw.Emit()
|
|
|
|
forecasted := []query.FloatPoint{
|
|
{Time: 1, Value: 3.93},
|
|
{Time: 2, Value: 4.957405463559748},
|
|
{Time: 3, Value: 7.012210102535647},
|
|
{Time: 4, Value: 10.099589257439924},
|
|
{Time: 5, Value: 14.229926188104242},
|
|
{Time: 6, Value: 19.418878968703797},
|
|
{Time: 7, Value: 25.68749172281409},
|
|
{Time: 8, Value: 33.062351305731305},
|
|
{Time: 9, Value: 41.575791076125206},
|
|
{Time: 10, Value: 51.26614395589263},
|
|
{Time: 11, Value: 62.178047564264595},
|
|
{Time: 12, Value: 74.36280483872488},
|
|
{Time: 13, Value: 87.87880423073163},
|
|
{Time: 14, Value: 102.79200429905801},
|
|
{Time: 15, Value: 119.17648832929542},
|
|
{Time: 16, Value: 137.11509549747296},
|
|
{Time: 17, Value: 156.70013608313175},
|
|
{Time: 18, Value: 178.03419933863566},
|
|
{Time: 19, Value: 201.23106385518594},
|
|
{Time: 20, Value: 226.4167216525905},
|
|
{Time: 21, Value: 253.73052878285205},
|
|
{Time: 22, Value: 283.32649700397553},
|
|
{Time: 23, Value: 315.37474308085984},
|
|
{Time: 24, Value: 350.06311454009256},
|
|
{Time: 25, Value: 387.59901328556873},
|
|
{Time: 26, Value: 428.21144141893404},
|
|
{Time: 27, Value: 472.1532969569147},
|
|
{Time: 28, Value: 519.7039509590035},
|
|
{Time: 29, Value: 571.1721419458248},
|
|
}
|
|
|
|
if exp, got := len(forecasted), len(points); exp != got {
|
|
t.Fatalf("unexpected number of points emitted: got %d exp %d", got, exp)
|
|
}
|
|
for i := range forecasted {
|
|
if exp, got := forecasted[i].Time, points[i].Time; got != exp {
|
|
t.Errorf("unexpected time on points[%d] got %v exp %v", i, got, exp)
|
|
}
|
|
if exp, got := forecasted[i].Value, points[i].Value; !almostEqual(got, exp) {
|
|
t.Errorf("unexpected value on points[%d] got %v exp %v", i, got, exp)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestHoltWinters_USPopulation_Missing(t *testing.T) {
|
|
series := []query.FloatPoint{
|
|
{Time: 1, Value: 3.93},
|
|
{Time: 2, Value: 5.31},
|
|
{Time: 3, Value: 7.24},
|
|
{Time: 4, Value: 9.64},
|
|
{Time: 5, Value: 12.90},
|
|
{Time: 6, Value: 17.10},
|
|
{Time: 7, Value: 23.20},
|
|
{Time: 8, Value: 31.40},
|
|
{Time: 10, Value: 50.20},
|
|
{Time: 11, Value: 62.90},
|
|
{Time: 12, Value: 76.00},
|
|
{Time: 13, Value: 92.00},
|
|
{Time: 15, Value: 122.80},
|
|
{Time: 16, Value: 131.70},
|
|
{Time: 17, Value: 151.30},
|
|
{Time: 19, Value: 203.20},
|
|
}
|
|
hw := query.NewFloatHoltWintersReducer(10, 0, true, 1)
|
|
for _, p := range series {
|
|
hw.AggregateFloat(&p)
|
|
}
|
|
points := hw.Emit()
|
|
|
|
forecasted := []query.FloatPoint{
|
|
{Time: 1, Value: 3.93},
|
|
{Time: 2, Value: 4.8931364428135105},
|
|
{Time: 3, Value: 6.962653629047061},
|
|
{Time: 4, Value: 10.056207765903274},
|
|
{Time: 5, Value: 14.18435088129532},
|
|
{Time: 6, Value: 19.362939306110846},
|
|
{Time: 7, Value: 25.613247940326584},
|
|
{Time: 8, Value: 32.96213087008264},
|
|
{Time: 9, Value: 41.442230043017204},
|
|
{Time: 10, Value: 51.09223428526052},
|
|
{Time: 11, Value: 61.95719155158485},
|
|
{Time: 12, Value: 74.08887794968567},
|
|
{Time: 13, Value: 87.54622778052787},
|
|
{Time: 14, Value: 102.39582960014131},
|
|
{Time: 15, Value: 118.7124941463221},
|
|
{Time: 16, Value: 136.57990089987464},
|
|
{Time: 17, Value: 156.09133107941278},
|
|
{Time: 18, Value: 177.35049601833734},
|
|
{Time: 19, Value: 200.472471161683},
|
|
{Time: 20, Value: 225.58474737097785},
|
|
{Time: 21, Value: 252.82841286206823},
|
|
{Time: 22, Value: 282.35948095261017},
|
|
{Time: 23, Value: 314.3503808953992},
|
|
{Time: 24, Value: 348.99163145856954},
|
|
{Time: 25, Value: 386.49371962730555},
|
|
{Time: 26, Value: 427.08920989407727},
|
|
{Time: 27, Value: 471.0351131332573},
|
|
{Time: 28, Value: 518.615548088049},
|
|
{Time: 29, Value: 570.1447331101863},
|
|
}
|
|
|
|
if exp, got := len(forecasted), len(points); exp != got {
|
|
t.Fatalf("unexpected number of points emitted: got %d exp %d", got, exp)
|
|
}
|
|
for i := range forecasted {
|
|
if exp, got := forecasted[i].Time, points[i].Time; got != exp {
|
|
t.Errorf("unexpected time on points[%d] got %v exp %v", i, got, exp)
|
|
}
|
|
if exp, got := forecasted[i].Value, points[i].Value; !almostEqual(got, exp) {
|
|
t.Errorf("unexpected value on points[%d] got %v exp %v", i, got, exp)
|
|
}
|
|
}
|
|
}
|
|
func TestHoltWinters_RoundTime(t *testing.T) {
|
|
maxTime := time.Unix(0, influxql.MaxTime).Round(time.Second).UnixNano()
|
|
data := []query.FloatPoint{
|
|
{Time: maxTime - int64(5*time.Second), Value: 1},
|
|
{Time: maxTime - int64(4*time.Second+103*time.Millisecond), Value: 10},
|
|
{Time: maxTime - int64(3*time.Second+223*time.Millisecond), Value: 2},
|
|
{Time: maxTime - int64(2*time.Second+481*time.Millisecond), Value: 11},
|
|
}
|
|
hw := query.NewFloatHoltWintersReducer(2, 2, true, time.Second)
|
|
for _, p := range data {
|
|
hw.AggregateFloat(&p)
|
|
}
|
|
points := hw.Emit()
|
|
|
|
forecasted := []query.FloatPoint{
|
|
{Time: maxTime - int64(5*time.Second), Value: 1},
|
|
{Time: maxTime - int64(4*time.Second), Value: 10.006729104838234},
|
|
{Time: maxTime - int64(3*time.Second), Value: 1.998341814469269},
|
|
{Time: maxTime - int64(2*time.Second), Value: 10.997858830631172},
|
|
{Time: maxTime - int64(1*time.Second), Value: 4.085860238030013},
|
|
{Time: maxTime - int64(0*time.Second), Value: 11.35713604403339},
|
|
}
|
|
|
|
if exp, got := len(forecasted), len(points); exp != got {
|
|
t.Fatalf("unexpected number of points emitted: got %d exp %d", got, exp)
|
|
}
|
|
for i := range forecasted {
|
|
if exp, got := forecasted[i].Time, points[i].Time; got != exp {
|
|
t.Errorf("unexpected time on points[%d] got %v exp %v", i, got, exp)
|
|
}
|
|
if exp, got := forecasted[i].Value, points[i].Value; !almostEqual(got, exp) {
|
|
t.Errorf("unexpected value on points[%d] got %v exp %v", i, got, exp)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestHoltWinters_MaxTime(t *testing.T) {
|
|
data := []query.FloatPoint{
|
|
{Time: influxql.MaxTime - 1, Value: 1},
|
|
{Time: influxql.MaxTime, Value: 2},
|
|
}
|
|
hw := query.NewFloatHoltWintersReducer(1, 0, true, 1)
|
|
for _, p := range data {
|
|
hw.AggregateFloat(&p)
|
|
}
|
|
points := hw.Emit()
|
|
|
|
forecasted := []query.FloatPoint{
|
|
{Time: influxql.MaxTime - 1, Value: 1},
|
|
{Time: influxql.MaxTime, Value: 2.001516944066403},
|
|
{Time: influxql.MaxTime + 1, Value: 2.5365248972488343},
|
|
}
|
|
|
|
if exp, got := len(forecasted), len(points); exp != got {
|
|
t.Fatalf("unexpected number of points emitted: got %d exp %d", got, exp)
|
|
}
|
|
for i := range forecasted {
|
|
if exp, got := forecasted[i].Time, points[i].Time; got != exp {
|
|
t.Errorf("unexpected time on points[%d] got %v exp %v", i, got, exp)
|
|
}
|
|
if exp, got := forecasted[i].Value, points[i].Value; !almostEqual(got, exp) {
|
|
t.Errorf("unexpected value on points[%d] got %v exp %v", i, got, exp)
|
|
}
|
|
}
|
|
}
|
|
|
|
// TestSample_AllSamplesSeen attempts to verify that it is possible
|
|
// to get every subsample in a reasonable number of iterations.
|
|
//
|
|
// The idea here is that 30 iterations should be enough to hit every possible
|
|
// sequence at least once.
|
|
func TestSample_AllSamplesSeen(t *testing.T) {
|
|
ps := []query.FloatPoint{
|
|
{Time: 1, Value: 1},
|
|
{Time: 2, Value: 2},
|
|
{Time: 3, Value: 3},
|
|
}
|
|
|
|
// List of all the possible subsamples
|
|
samples := [][]query.FloatPoint{
|
|
{
|
|
{Time: 1, Value: 1},
|
|
{Time: 2, Value: 2},
|
|
},
|
|
{
|
|
{Time: 1, Value: 1},
|
|
{Time: 3, Value: 3},
|
|
},
|
|
{
|
|
{Time: 2, Value: 2},
|
|
{Time: 3, Value: 3},
|
|
},
|
|
}
|
|
|
|
// 30 iterations should be sufficient to guarantee that
|
|
// we hit every possible subsample.
|
|
for i := 0; i < 30; i++ {
|
|
s := query.NewFloatSampleReducer(2)
|
|
for _, p := range ps {
|
|
s.AggregateFloat(&p)
|
|
}
|
|
|
|
points := s.Emit()
|
|
|
|
for i, sample := range samples {
|
|
// if we find a sample that it matches, remove it from
|
|
// this list of possible samples
|
|
if deep.Equal(sample, points) {
|
|
samples = append(samples[:i], samples[i+1:]...)
|
|
break
|
|
}
|
|
}
|
|
|
|
// if samples is empty we've seen every sample, so we're done
|
|
if len(samples) == 0 {
|
|
return
|
|
}
|
|
|
|
// The FloatSampleReducer is seeded with time.Now().UnixNano(), and without this sleep,
|
|
// this test will fail on machines where UnixNano doesn't return full resolution.
|
|
// Specifically, some Windows machines will only return timestamps accurate to 100ns.
|
|
// While iterating through this test without an explicit sleep,
|
|
// we would only see one or two unique seeds across all the calls to NewFloatSampleReducer.
|
|
time.Sleep(time.Millisecond)
|
|
}
|
|
|
|
// If we missed a sample, report the error
|
|
if len(samples) != 0 {
|
|
t.Fatalf("expected all samples to be seen; unseen samples: %#v", samples)
|
|
}
|
|
}
|
|
|
|
func TestSample_SampleSizeLessThanNumPoints(t *testing.T) {
|
|
s := query.NewFloatSampleReducer(2)
|
|
|
|
ps := []query.FloatPoint{
|
|
{Time: 1, Value: 1},
|
|
{Time: 2, Value: 2},
|
|
{Time: 3, Value: 3},
|
|
}
|
|
|
|
for _, p := range ps {
|
|
s.AggregateFloat(&p)
|
|
}
|
|
|
|
points := s.Emit()
|
|
|
|
if exp, got := 2, len(points); exp != got {
|
|
t.Fatalf("unexpected number of points emitted: got %d exp %d", got, exp)
|
|
}
|
|
}
|
|
|
|
func TestSample_SampleSizeGreaterThanNumPoints(t *testing.T) {
|
|
s := query.NewFloatSampleReducer(4)
|
|
|
|
ps := []query.FloatPoint{
|
|
{Time: 1, Value: 1},
|
|
{Time: 2, Value: 2},
|
|
{Time: 3, Value: 3},
|
|
}
|
|
|
|
for _, p := range ps {
|
|
s.AggregateFloat(&p)
|
|
}
|
|
|
|
points := s.Emit()
|
|
|
|
if exp, got := len(ps), len(points); exp != got {
|
|
t.Fatalf("unexpected number of points emitted: got %d exp %d", got, exp)
|
|
}
|
|
|
|
if !deep.Equal(ps, points) {
|
|
t.Fatalf("unexpected points: %s", spew.Sdump(points))
|
|
}
|
|
}
|
|
|
|
func TestHll_SumAndMergeHll(t *testing.T) {
|
|
assert := tassert.New(t)
|
|
require := trequire.New(t)
|
|
|
|
// Make 3000 random strings
|
|
r := rand.New(rand.NewSource(42))
|
|
input := make([]*query.StringPoint, 0, 3000)
|
|
for i := 0; i < 3000; i++ {
|
|
input = append(input, &query.StringPoint{Value: strconv.FormatUint(r.Uint64(), 10)})
|
|
}
|
|
|
|
// Insert overlapping sections of the same points array to different reducers
|
|
s1 := query.NewStringSumHllReducer()
|
|
for _, p := range input[:2000] {
|
|
s1.AggregateString(p)
|
|
}
|
|
point1 := s1.Emit()
|
|
s2 := query.NewStringSumHllReducer()
|
|
for _, p := range input[1000:] {
|
|
s2.AggregateString(p)
|
|
}
|
|
point2 := s2.Emit()
|
|
// Demonstration of the input: repeatably seeded pseudorandom
|
|
// stringified integers (so we are testing the counting of unique strings,
|
|
// not unique integers).
|
|
require.Equal("17190211103962133664", input[2999].Value)
|
|
|
|
checkStringFingerprint := func(prefix string, length int, hash string, check string) {
|
|
assert.Equal(length, len(check))
|
|
assert.Equal(prefix, check[:len(prefix)])
|
|
h := sha1.New()
|
|
h.Write([]byte(check))
|
|
assert.Equal(hash, fmt.Sprintf("%x", h.Sum(nil)))
|
|
}
|
|
|
|
require.Equal(len(point1), 1)
|
|
require.Equal(len(point2), 1)
|
|
checkStringFingerprint("HLL_AhABAAAAAAAAB9BIDQAJAAAUUaKsA4K/AtARkuMBsJwEyp8O",
|
|
6964, "c59fa799fe8e78ab5347de385bf2a7c5b8085882", point1[0].Value)
|
|
checkStringFingerprint("HLL_AhABAAAAAAAAB9Db0QAHAAAUaP6aAaSRAoK/Ap70B/xSysEE",
|
|
6996, "5f1696dfb455baab7fdb56ffd2197d27b09d6dcf", point2[0].Value)
|
|
|
|
m := query.NewStringMergeHllReducer()
|
|
m.AggregateString(&point1[0])
|
|
m.AggregateString(&point2[0])
|
|
merged := m.Emit()
|
|
require.Equal(1, len(merged))
|
|
checkStringFingerprint("HLL_AhAAAAEAAAAAAAAAAAAAAAAAAAAAAAAAAAACAAAAAAAAAAAA",
|
|
87396, "e5320860aa322efe9af268e171df916d2186c75f", merged[0].Value)
|
|
|
|
m.AggregateString(&query.StringPoint{
|
|
Time: query.ZeroTime,
|
|
Value: "some random string",
|
|
})
|
|
mergedError := m.Emit()
|
|
// mid-level errors are:
|
|
require.Equal(1, len(mergedError))
|
|
assert.Equal("HLLERROR Bad prefix for hll.Plus", mergedError[0].Value)
|
|
|
|
c := query.NewCountHllReducer()
|
|
c.AggregateString(&merged[0])
|
|
counted := c.Emit()
|
|
require.Equal(1, len(counted))
|
|
// Counted 4000 points, 3000 distinct points, answer is 2994 ≈ 3000
|
|
assert.Equal(uint64(2994), counted[0].Value)
|
|
|
|
c.AggregateString(&query.StringPoint{
|
|
Time: query.ZeroTime,
|
|
Value: "HLLERROR Bad prefix for hll.Plus",
|
|
})
|
|
counted = c.Emit()
|
|
require.Equal(1, len(counted))
|
|
// When we hit marshal/unmarshal errors
|
|
assert.Equal(uint64(0), counted[0].Value)
|
|
}
|