blob: 5158a159ae244b82c00d79763f374e23332a01ea [file] [log] [blame] [edit]
/*
* Copyright 2019 Dgraph Labs, Inc. and Contributors
*
* 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 table
import (
"fmt"
"math/rand"
"os"
"testing"
"time"
"github.com/stretchr/testify/require"
"github.com/dgraph-io/badger/v2/options"
"github.com/dgraph-io/badger/v2/pb"
"github.com/dgraph-io/badger/v2/y"
)
func TestTableIndex(t *testing.T) {
rand.Seed(time.Now().Unix())
keysCount := 100000
key := make([]byte, 32)
_, err := rand.Read(key)
require.NoError(t, err)
subTest := []struct {
name string
opts Options
}{
{
name: "No encyption/compression",
opts: Options{
BlockSize: 4 * 1024,
BloomFalsePositive: 0.01,
TableSize: 30 << 20,
},
},
{
// Encryption mode.
name: "Only encryption",
opts: Options{
BlockSize: 4 * 1024,
BloomFalsePositive: 0.01,
TableSize: 30 << 20,
DataKey: &pb.DataKey{Data: key},
},
},
{
// Compression mode.
name: "Only compression",
opts: Options{
BlockSize: 4 * 1024,
BloomFalsePositive: 0.01,
TableSize: 30 << 20,
Compression: options.ZSTD,
ZSTDCompressionLevel: 3,
},
},
{
// Compression mode and encryption.
name: "Compression and encryption",
opts: Options{
BlockSize: 4 * 1024,
BloomFalsePositive: 0.01,
TableSize: 30 << 20,
Compression: options.ZSTD,
ZSTDCompressionLevel: 3,
DataKey: &pb.DataKey{Data: key},
},
},
}
for _, tt := range subTest {
t.Run(tt.name, func(t *testing.T) {
opt := tt.opts
builder := NewTableBuilder(opt)
filename := fmt.Sprintf("%s%c%d.sst", os.TempDir(), os.PathSeparator, rand.Uint32())
f, err := y.OpenSyncedFile(filename, true)
require.NoError(t, err)
blockFirstKeys := make([][]byte, 0)
blockCount := 0
for i := 0; i < keysCount; i++ {
k := []byte(fmt.Sprintf("%016x", i))
v := fmt.Sprintf("%d", i)
vs := y.ValueStruct{Value: []byte(v)}
if i == 0 { // This is first key for first block.
blockFirstKeys = append(blockFirstKeys, k)
blockCount = 1
} else if builder.shouldFinishBlock(k, vs) {
blockCount++
blockFirstKeys = append(blockFirstKeys, k)
}
builder.Add(k, vs, 0)
}
_, err = f.Write(builder.Finish())
require.NoError(t, err, "unable to write to file")
tbl, err := OpenTable(f, opt)
require.NoError(t, err, "unable to open table")
if opt.DataKey == nil {
// key id is zero if there is no datakey.
require.Equal(t, tbl.KeyID(), uint64(0))
}
// Ensure index is built correctly
require.Equal(t, blockCount, len(tbl.blockIndex))
for i, ko := range tbl.blockIndex {
require.Equal(t, ko.Key, blockFirstKeys[i])
}
f.Close()
require.NoError(t, os.RemoveAll(filename))
})
}
}
func TestInvalidCompression(t *testing.T) {
keyPrefix := "key"
opts := Options{Compression: options.ZSTD}
f := buildTestTable(t, keyPrefix, 1000, opts)
t.Run("with correct decompression algo", func(t *testing.T) {
_, err := OpenTable(f, opts)
require.NoError(t, err)
})
t.Run("with incorrect decompression algo", func(t *testing.T) {
// Set incorrect compression algorithm.
opts.Compression = options.Snappy
_, err := OpenTable(f, opts)
require.Error(t, err)
})
}
func BenchmarkBuilder(b *testing.B) {
rand.Seed(time.Now().Unix())
key := func(i int) []byte {
return []byte(fmt.Sprintf("%032d", i))
}
val := make([]byte, 32)
rand.Read(val)
vs := y.ValueStruct{Value: []byte(val)}
keysCount := 1300000 // This number of entries consumes ~64MB of memory.
var keyList [][]byte
for i := 0; i < keysCount; i++ {
keyList = append(keyList, key(i))
}
bench := func(b *testing.B, opt *Options) {
b.SetBytes(int64(keysCount) * (32 + 32))
opt.BlockSize = 4 * 1024
opt.BloomFalsePositive = 0.01
opt.TableSize = 5 << 20
b.ResetTimer()
for i := 0; i < b.N; i++ {
builder := NewTableBuilder(*opt)
for j := 0; j < keysCount; j++ {
builder.Add(keyList[j], vs, 0)
}
_ = builder.Finish()
}
}
b.Run("no compression", func(b *testing.B) {
var opt Options
opt.Compression = options.None
bench(b, &opt)
})
b.Run("encryption", func(b *testing.B) {
var opt Options
key := make([]byte, 32)
rand.Read(key)
opt.DataKey = &pb.DataKey{Data: key}
bench(b, &opt)
})
b.Run("zstd compression", func(b *testing.B) {
var opt Options
opt.Compression = options.ZSTD
b.Run("level 1", func(b *testing.B) {
opt.ZSTDCompressionLevel = 1
bench(b, &opt)
})
b.Run("level 3", func(b *testing.B) {
opt.ZSTDCompressionLevel = 3
bench(b, &opt)
})
b.Run("level 15", func(b *testing.B) {
opt.ZSTDCompressionLevel = 15
bench(b, &opt)
})
})
}