mirror of
https://github.com/VictoriaMetrics/VictoriaMetrics.git
synced 2024-11-21 14:44:00 +00:00
36a1fdca6c
This allows consistently using errors.Is() for verifying whether the given error wraps some other known error.
150 lines
5.3 KiB
Go
150 lines
5.3 KiB
Go
package storage
|
|
|
|
import (
|
|
"fmt"
|
|
"math/rand"
|
|
"reflect"
|
|
"strings"
|
|
"testing"
|
|
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/lib/encoding"
|
|
)
|
|
|
|
func TestBlockMarshalUnmarshalPortable(t *testing.T) {
|
|
rng := rand.New(rand.NewSource(1))
|
|
var b Block
|
|
for i := 0; i < 1000; i++ {
|
|
b.Reset()
|
|
rowsCount := rng.Intn(maxRowsPerBlock) + 1
|
|
b.timestamps = getRandTimestamps(rowsCount)
|
|
b.values = getRandValues(rowsCount)
|
|
b.bh.Scale = int16(rng.Intn(30) - 15)
|
|
b.bh.PrecisionBits = uint8(64 - (i % 64))
|
|
testBlockMarshalUnmarshalPortable(t, &b)
|
|
}
|
|
}
|
|
|
|
func testBlockMarshalUnmarshalPortable(t *testing.T, b *Block) {
|
|
var b1, b2 Block
|
|
rowsCount := len(b.values)
|
|
b1.CopyFrom(b)
|
|
data := b1.MarshalPortable(nil)
|
|
if b1.bh.RowsCount != uint32(rowsCount) {
|
|
t.Fatalf("unexpected number of rows marshaled; got %d; want %d", b1.bh.RowsCount, rowsCount)
|
|
}
|
|
tail, err := b2.UnmarshalPortable(data)
|
|
if err != nil {
|
|
t.Fatalf("unexpected error: %s", err)
|
|
}
|
|
if len(tail) > 0 {
|
|
t.Fatalf("unexpected non-empty tail: %X", tail)
|
|
}
|
|
compareBlocksPortable(t, &b2, b, &b1.bh)
|
|
|
|
// Verify non-empty prefix and suffix
|
|
prefix := "prefix"
|
|
suffix := "suffix"
|
|
data = append(data[:0], prefix...)
|
|
data = b1.MarshalPortable(data)
|
|
if b1.bh.RowsCount != uint32(rowsCount) {
|
|
t.Fatalf("unexpected number of rows marshaled; got %d; want %d", b1.bh.RowsCount, rowsCount)
|
|
}
|
|
if !strings.HasPrefix(string(data), prefix) {
|
|
t.Fatalf("unexpected prefix in %X; want %X", data, prefix)
|
|
}
|
|
data = data[len(prefix):]
|
|
data = append(data, suffix...)
|
|
tail, err = b2.UnmarshalPortable(data)
|
|
if err != nil {
|
|
t.Fatalf("unexpected error: %s", err)
|
|
}
|
|
if string(tail) != suffix {
|
|
t.Fatalf("unexpected tail; got %X; want %X", tail, suffix)
|
|
}
|
|
compareBlocksPortable(t, &b2, b, &b1.bh)
|
|
}
|
|
|
|
func compareBlocksPortable(t *testing.T, b1, bExpected *Block, bhExpected *blockHeader) {
|
|
t.Helper()
|
|
if b1.bh.MinTimestamp != bhExpected.MinTimestamp {
|
|
t.Fatalf("unexpected MinTimestamp; got %d; want %d", b1.bh.MinTimestamp, bhExpected.MinTimestamp)
|
|
}
|
|
if b1.bh.MaxTimestamp != bhExpected.MaxTimestamp {
|
|
t.Fatalf("unexpected MinTimestamp; got %d; want %d", b1.bh.MaxTimestamp, bhExpected.MaxTimestamp)
|
|
}
|
|
if b1.bh.FirstValue != bhExpected.FirstValue {
|
|
t.Fatalf("unexpected FirstValue; got %d; want %d", b1.bh.FirstValue, bhExpected.FirstValue)
|
|
}
|
|
if b1.bh.RowsCount != bhExpected.RowsCount {
|
|
t.Fatalf("unexpected RowsCount; got %d; want %d", b1.bh.RowsCount, bhExpected.RowsCount)
|
|
}
|
|
if b1.bh.Scale != bhExpected.Scale {
|
|
t.Fatalf("unexpected Scale; got %d; want %d", b1.bh.Scale, bhExpected.Scale)
|
|
}
|
|
if b1.bh.TimestampsMarshalType != bhExpected.TimestampsMarshalType {
|
|
t.Fatalf("unexpected TimestampsMarshalType; got %d; want %d", b1.bh.TimestampsMarshalType, bhExpected.TimestampsMarshalType)
|
|
}
|
|
if b1.bh.ValuesMarshalType != bhExpected.ValuesMarshalType {
|
|
t.Fatalf("unexpected ValuesMarshalType; got %d; want %d", b1.bh.ValuesMarshalType, bhExpected.ValuesMarshalType)
|
|
}
|
|
if b1.bh.PrecisionBits != bhExpected.PrecisionBits {
|
|
t.Fatalf("unexpected PrecisionBits; got %d; want %d", b1.bh.PrecisionBits, bhExpected.PrecisionBits)
|
|
}
|
|
|
|
timestampsExpected := getTimestampsForPrecisionBits(bExpected.timestamps, bhExpected.PrecisionBits)
|
|
valuesExpected := getValuesForPrecisionBits(bExpected.values, bhExpected.PrecisionBits)
|
|
|
|
if !reflect.DeepEqual(b1.values, valuesExpected) {
|
|
t.Fatalf("unexpected values for precisionBits=%d; got\n%d\nwant\n%d", b1.bh.PrecisionBits, b1.values, valuesExpected)
|
|
}
|
|
if !reflect.DeepEqual(b1.timestamps, timestampsExpected) {
|
|
t.Fatalf("unexpected timestamps for precisionBits=%d; got\n%d\nwant\n%d", b1.bh.PrecisionBits, b1.timestamps, timestampsExpected)
|
|
}
|
|
if len(b1.values) != int(bhExpected.RowsCount) {
|
|
t.Fatalf("unexpected number of values; got %d; want %d", len(b1.values), bhExpected.RowsCount)
|
|
}
|
|
if len(b1.timestamps) != int(bhExpected.RowsCount) {
|
|
t.Fatalf("unexpected number of timestamps; got %d; want %d", len(b1.timestamps), bhExpected.RowsCount)
|
|
}
|
|
}
|
|
|
|
func getTimestampsForPrecisionBits(timestamps []int64, precisionBits uint8) []int64 {
|
|
data, marshalType, firstTimestamp := encoding.MarshalTimestamps(nil, timestamps, precisionBits)
|
|
timestampsAdjusted, err := encoding.UnmarshalTimestamps(nil, data, marshalType, firstTimestamp, len(timestamps))
|
|
if err != nil {
|
|
panic(fmt.Errorf("BUG: cannot unmarshal timestamps with precisionBits %d: %w", precisionBits, err))
|
|
}
|
|
minTimestamp := timestamps[0]
|
|
maxTimestamp := timestamps[len(timestamps)-1]
|
|
encoding.EnsureNonDecreasingSequence(timestampsAdjusted, minTimestamp, maxTimestamp)
|
|
return timestampsAdjusted
|
|
}
|
|
|
|
func getValuesForPrecisionBits(values []int64, precisionBits uint8) []int64 {
|
|
data, marshalType, firstValue := encoding.MarshalValues(nil, values, precisionBits)
|
|
valuesAdjusted, err := encoding.UnmarshalValues(nil, data, marshalType, firstValue, len(values))
|
|
if err != nil {
|
|
panic(fmt.Errorf("BUG: cannot unmarshal values with precisionBits %d: %w", precisionBits, err))
|
|
}
|
|
return valuesAdjusted
|
|
}
|
|
|
|
func getRandValues(rowsCount int) []int64 {
|
|
rng := rand.New(rand.NewSource(1))
|
|
a := make([]int64, rowsCount)
|
|
for i := 0; i < rowsCount; i++ {
|
|
a[i] = int64(rng.Intn(1e5) - 0.5e5)
|
|
}
|
|
return a
|
|
}
|
|
|
|
func getRandTimestamps(rowsCount int) []int64 {
|
|
rng := rand.New(rand.NewSource(1))
|
|
a := make([]int64, rowsCount)
|
|
ts := int64(rng.Intn(1e9))
|
|
for i := 0; i < rowsCount; i++ {
|
|
a[i] = ts
|
|
ts += int64(rng.Intn(1e5))
|
|
}
|
|
return a
|
|
}
|