mirror of https://github.com/prometheus/prometheus
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
311 lines
6.5 KiB
311 lines
6.5 KiB
// Copyright 2017 The Prometheus Authors |
|
// Licensed under the Apache License, Version 2.0 (the "License"); |
|
// you may not use this file except in compliance with the License. |
|
// You may obtain a copy of the License at |
|
// |
|
// http://www.apache.org/licenses/LICENSE-2.0 |
|
// |
|
// Unless required by applicable law or agreed to in writing, software |
|
// distributed under the License is distributed on an "AS IS" BASIS, |
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
|
// See the License for the specific language governing permissions and |
|
// limitations under the License. |
|
|
|
package chunkenc |
|
|
|
import ( |
|
"errors" |
|
"fmt" |
|
"io" |
|
"math/rand" |
|
"testing" |
|
|
|
"github.com/stretchr/testify/require" |
|
) |
|
|
|
type pair struct { |
|
t int64 |
|
v float64 |
|
} |
|
|
|
func TestChunk(t *testing.T) { |
|
for enc, nc := range map[Encoding]func() Chunk{ |
|
EncXOR: func() Chunk { return NewXORChunk() }, |
|
} { |
|
t.Run(fmt.Sprintf("%v", enc), func(t *testing.T) { |
|
for range make([]struct{}, 1) { |
|
c := nc() |
|
testChunk(t, c) |
|
} |
|
}) |
|
} |
|
} |
|
|
|
func testChunk(t *testing.T, c Chunk) { |
|
app, err := c.Appender() |
|
require.NoError(t, err) |
|
|
|
var exp []pair |
|
var ( |
|
ts = int64(1234123324) |
|
v = 1243535.123 |
|
) |
|
for i := 0; i < 300; i++ { |
|
ts += int64(rand.Intn(10000) + 1) |
|
if i%2 == 0 { |
|
v += float64(rand.Intn(1000000)) |
|
} else { |
|
v -= float64(rand.Intn(1000000)) |
|
} |
|
|
|
// Start with a new appender every 10th sample. This emulates starting |
|
// appending to a partially filled chunk. |
|
if i%10 == 0 { |
|
app, err = c.Appender() |
|
require.NoError(t, err) |
|
} |
|
|
|
app.Append(ts, v) |
|
exp = append(exp, pair{t: ts, v: v}) |
|
} |
|
|
|
// 1. Expand iterator in simple case. |
|
it1 := c.Iterator(nil) |
|
var res1 []pair |
|
for it1.Next() == ValFloat { |
|
ts, v := it1.At() |
|
res1 = append(res1, pair{t: ts, v: v}) |
|
} |
|
require.NoError(t, it1.Err()) |
|
require.Equal(t, exp, res1) |
|
|
|
// 2. Expand second iterator while reusing first one. |
|
it2 := c.Iterator(it1) |
|
var res2 []pair |
|
for it2.Next() == ValFloat { |
|
ts, v := it2.At() |
|
res2 = append(res2, pair{t: ts, v: v}) |
|
} |
|
require.NoError(t, it2.Err()) |
|
require.Equal(t, exp, res2) |
|
|
|
// 3. Test iterator Seek. |
|
mid := len(exp) / 2 |
|
|
|
it3 := c.Iterator(nil) |
|
var res3 []pair |
|
require.Equal(t, ValFloat, it3.Seek(exp[mid].t)) |
|
// Below ones should not matter. |
|
require.Equal(t, ValFloat, it3.Seek(exp[mid].t)) |
|
require.Equal(t, ValFloat, it3.Seek(exp[mid].t)) |
|
ts, v = it3.At() |
|
res3 = append(res3, pair{t: ts, v: v}) |
|
|
|
for it3.Next() == ValFloat { |
|
ts, v := it3.At() |
|
res3 = append(res3, pair{t: ts, v: v}) |
|
} |
|
require.NoError(t, it3.Err()) |
|
require.Equal(t, exp[mid:], res3) |
|
require.Equal(t, ValNone, it3.Seek(exp[len(exp)-1].t+1)) |
|
} |
|
|
|
func TestPool(t *testing.T) { |
|
p := NewPool() |
|
for _, tc := range []struct { |
|
name string |
|
encoding Encoding |
|
expErr error |
|
}{ |
|
{ |
|
name: "xor", |
|
encoding: EncXOR, |
|
}, |
|
{ |
|
name: "histogram", |
|
encoding: EncHistogram, |
|
}, |
|
{ |
|
name: "float histogram", |
|
encoding: EncFloatHistogram, |
|
}, |
|
{ |
|
name: "invalid encoding", |
|
encoding: EncNone, |
|
expErr: fmt.Errorf(`invalid chunk encoding "none"`), |
|
}, |
|
} { |
|
t.Run(tc.name, func(t *testing.T) { |
|
c, err := p.Get(tc.encoding, []byte("test")) |
|
if tc.expErr != nil { |
|
require.EqualError(t, err, tc.expErr.Error()) |
|
return |
|
} |
|
|
|
require.NoError(t, err) |
|
|
|
var b *bstream |
|
switch tc.encoding { |
|
case EncHistogram: |
|
b = &c.(*HistogramChunk).b |
|
case EncFloatHistogram: |
|
b = &c.(*FloatHistogramChunk).b |
|
default: |
|
b = &c.(*XORChunk).b |
|
} |
|
|
|
require.Equal(t, &bstream{ |
|
stream: []byte("test"), |
|
count: 0, |
|
}, b) |
|
|
|
b.count = 1 |
|
require.NoError(t, p.Put(c)) |
|
require.Equal(t, &bstream{ |
|
stream: nil, |
|
count: 0, |
|
}, b) |
|
}) |
|
} |
|
|
|
t.Run("put bad chunk wrapper", func(t *testing.T) { |
|
// When a wrapping chunk poses as an encoding it can't be converted to, Put should skip it. |
|
c := fakeChunk{ |
|
encoding: EncXOR, |
|
t: t, |
|
} |
|
require.NoError(t, p.Put(c)) |
|
}) |
|
t.Run("put invalid encoding", func(t *testing.T) { |
|
c := fakeChunk{ |
|
encoding: EncNone, |
|
t: t, |
|
} |
|
require.EqualError(t, p.Put(c), `invalid chunk encoding "none"`) |
|
}) |
|
} |
|
|
|
type fakeChunk struct { |
|
Chunk |
|
|
|
encoding Encoding |
|
t *testing.T |
|
} |
|
|
|
func (c fakeChunk) Encoding() Encoding { |
|
return c.encoding |
|
} |
|
|
|
func (c fakeChunk) Reset([]byte) { |
|
c.t.Fatal("Reset should not be called") |
|
} |
|
|
|
func benchmarkIterator(b *testing.B, newChunk func() Chunk) { |
|
const samplesPerChunk = 250 |
|
var ( |
|
t = int64(1234123324) |
|
v = 1243535.123 |
|
exp []pair |
|
) |
|
for i := 0; i < samplesPerChunk; i++ { |
|
// t += int64(rand.Intn(10000) + 1) |
|
t += int64(1000) |
|
// v = rand.Float64() |
|
v += float64(100) |
|
exp = append(exp, pair{t: t, v: v}) |
|
} |
|
|
|
chunk := newChunk() |
|
{ |
|
a, err := chunk.Appender() |
|
if err != nil { |
|
b.Fatalf("get appender: %s", err) |
|
} |
|
j := 0 |
|
for _, p := range exp { |
|
if j > 250 { |
|
break |
|
} |
|
a.Append(p.t, p.v) |
|
j++ |
|
} |
|
} |
|
|
|
b.ReportAllocs() |
|
b.ResetTimer() |
|
|
|
var res float64 |
|
var it Iterator |
|
for i := 0; i < b.N; { |
|
it := chunk.Iterator(it) |
|
|
|
for it.Next() == ValFloat { |
|
_, v := it.At() |
|
res = v |
|
i++ |
|
} |
|
if err := it.Err(); err != nil && !errors.Is(err, io.EOF) { |
|
require.NoError(b, err) |
|
} |
|
_ = res |
|
} |
|
} |
|
|
|
func newXORChunk() Chunk { |
|
return NewXORChunk() |
|
} |
|
|
|
func BenchmarkXORIterator(b *testing.B) { |
|
benchmarkIterator(b, newXORChunk) |
|
} |
|
|
|
func BenchmarkXORAppender(b *testing.B) { |
|
r := rand.New(rand.NewSource(1)) |
|
b.Run("constant", func(b *testing.B) { |
|
benchmarkAppender(b, func() (int64, float64) { |
|
return 1000, 0 |
|
}, newXORChunk) |
|
}) |
|
b.Run("random steps", func(b *testing.B) { |
|
benchmarkAppender(b, func() (int64, float64) { |
|
return int64(r.Intn(100) - 50 + 15000), // 15 seconds +- up to 100ms of jitter. |
|
float64(r.Intn(100) - 50) // Varying from -50 to +50 in 100 discrete steps. |
|
}, newXORChunk) |
|
}) |
|
b.Run("random 0-1", func(b *testing.B) { |
|
benchmarkAppender(b, func() (int64, float64) { |
|
return int64(r.Intn(100) - 50 + 15000), // 15 seconds +- up to 100ms of jitter. |
|
r.Float64() // Random between 0 and 1.0. |
|
}, newXORChunk) |
|
}) |
|
} |
|
|
|
func benchmarkAppender(b *testing.B, deltas func() (int64, float64), newChunk func() Chunk) { |
|
var ( |
|
t = int64(1234123324) |
|
v = 1243535.123 |
|
) |
|
const nSamples = 120 // Same as tsdb.DefaultSamplesPerChunk. |
|
var exp []pair |
|
for i := 0; i < nSamples; i++ { |
|
dt, dv := deltas() |
|
t += dt |
|
v += dv |
|
exp = append(exp, pair{t: t, v: v}) |
|
} |
|
|
|
b.ReportAllocs() |
|
b.ResetTimer() |
|
|
|
for i := 0; i < b.N; i++ { |
|
c := newChunk() |
|
|
|
a, err := c.Appender() |
|
if err != nil { |
|
b.Fatalf("get appender: %s", err) |
|
} |
|
for _, p := range exp { |
|
a.Append(p.t, p.v) |
|
} |
|
} |
|
}
|
|
|