package streamaggr import ( "fmt" "reflect" "sync" "testing" "time" ) func TestDedupAggrSerial(t *testing.T) { da := newDedupAggr(2) const seriesCount = 100_000 expectedSamplesMap := make(map[string]pushSample) for i := 0; i < 2; i++ { samples := make([]pushSample, seriesCount) for j := range samples { sample := &samples[j] sample.key = fmt.Sprintf("key_%d", j) sample.value = float64(i + j) expectedSamplesMap[sample.key] = *sample } data := &pushCtxData{ samples: samples, } da.pushSamples(data) } if n := da.sizeBytes(); n > 5_000_000 { t.Fatalf("too big dedupAggr state before flush: %d bytes; it shouldn't exceed 5_000_000 bytes", n) } if n := da.itemsCount(); n != seriesCount { t.Fatalf("unexpected itemsCount; got %d; want %d", n, seriesCount) } flushedSamplesMap := make(map[string]pushSample) var mu sync.Mutex flushSamples := func(ctx *pushCtxData) { mu.Lock() for _, sample := range ctx.samples { flushedSamplesMap[sample.key] = sample } mu.Unlock() } flushTimestamp := time.Now().UnixMilli() da.flush(flushSamples, flushTimestamp, 0, 0) if !reflect.DeepEqual(expectedSamplesMap, flushedSamplesMap) { t.Fatalf("unexpected samples;\ngot\n%v\nwant\n%v", flushedSamplesMap, expectedSamplesMap) } if n := da.sizeBytes(); n > 17_000 { t.Fatalf("too big dedupAggr state after flush; %d bytes; it shouldn't exceed 17_000 bytes", n) } if n := da.itemsCount(); n != 0 { t.Fatalf("unexpected non-zero itemsCount after flush; got %d", n) } } func TestDedupAggrConcurrent(_ *testing.T) { const concurrency = 5 const seriesCount = 10_000 da := newDedupAggr(2) var wg sync.WaitGroup for i := 0; i < concurrency; i++ { wg.Add(1) go func() { defer wg.Done() for i := 0; i < 10; i++ { data := &pushCtxData{ samples: make([]pushSample, seriesCount), } for j := range data.samples { sample := &data.samples[j] sample.key = fmt.Sprintf("key_%d", j) sample.value = float64(i + j) } da.pushSamples(data) } }() } wg.Wait() }