mirror of
https://source.quilibrium.com/quilibrium/ceremonyclient.git
synced 2025-01-23 14:15:18 +00:00
326 lines
8.0 KiB
Go
326 lines
8.0 KiB
Go
|
// Copyright 2018 The LevelDB-Go and Pebble Authors. All rights reserved. Use
|
||
|
// of this source code is governed by a BSD-style license that can be found in
|
||
|
// the LICENSE file.
|
||
|
|
||
|
package pebble
|
||
|
|
||
|
import (
|
||
|
"fmt"
|
||
|
"math/rand"
|
||
|
"runtime"
|
||
|
"testing"
|
||
|
"time"
|
||
|
|
||
|
"github.com/cockroachdb/errors"
|
||
|
"github.com/cockroachdb/pebble/internal/base"
|
||
|
"github.com/cockroachdb/pebble/vfs"
|
||
|
"github.com/stretchr/testify/require"
|
||
|
)
|
||
|
|
||
|
// testingRandomized randomizes some default options. Currently, it's
|
||
|
// used for testing under a random format major version in some tests.
|
||
|
func (o *Options) testingRandomized(t testing.TB) *Options {
|
||
|
if o == nil {
|
||
|
o = &Options{}
|
||
|
}
|
||
|
if o.Logger == nil {
|
||
|
o.Logger = testLogger{t: t}
|
||
|
}
|
||
|
if o.FormatMajorVersion == FormatDefault {
|
||
|
// Pick a random format major version from the range
|
||
|
// [MostCompatible, FormatNewest].
|
||
|
o.FormatMajorVersion = FormatMajorVersion(rand.Intn(int(internalFormatNewest)) + 1)
|
||
|
t.Logf("Running %s with format major version %s", t.Name(), o.FormatMajorVersion.String())
|
||
|
}
|
||
|
return o
|
||
|
}
|
||
|
|
||
|
func testingRandomized(t testing.TB, o *Options) *Options {
|
||
|
o.testingRandomized(t)
|
||
|
return o
|
||
|
}
|
||
|
|
||
|
func TestLevelOptions(t *testing.T) {
|
||
|
var opts *Options
|
||
|
opts = opts.EnsureDefaults()
|
||
|
|
||
|
testCases := []struct {
|
||
|
level int
|
||
|
targetFileSize int64
|
||
|
}{
|
||
|
{0, 2 << 20},
|
||
|
{1, (2 * 2) << 20},
|
||
|
{2, (4 * 2) << 20},
|
||
|
{3, (8 * 2) << 20},
|
||
|
{4, (16 * 2) << 20},
|
||
|
{5, (32 * 2) << 20},
|
||
|
{6, (64 * 2) << 20},
|
||
|
}
|
||
|
for _, c := range testCases {
|
||
|
l := opts.Level(c.level)
|
||
|
if c.targetFileSize != l.TargetFileSize {
|
||
|
t.Fatalf("%d: expected target-file-size %d, but found %d",
|
||
|
c.level, c.targetFileSize, l.TargetFileSize)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func TestOptionsString(t *testing.T) {
|
||
|
n := runtime.GOMAXPROCS(8)
|
||
|
defer runtime.GOMAXPROCS(n)
|
||
|
|
||
|
const expected = `[Version]
|
||
|
pebble_version=0.1
|
||
|
|
||
|
[Options]
|
||
|
bytes_per_sync=524288
|
||
|
cache_size=8388608
|
||
|
cleaner=delete
|
||
|
compaction_debt_concurrency=1073741824
|
||
|
comparer=leveldb.BytewiseComparator
|
||
|
disable_wal=false
|
||
|
flush_delay_delete_range=0s
|
||
|
flush_delay_range_key=0s
|
||
|
flush_split_bytes=4194304
|
||
|
format_major_version=1
|
||
|
l0_compaction_concurrency=10
|
||
|
l0_compaction_file_threshold=500
|
||
|
l0_compaction_threshold=4
|
||
|
l0_stop_writes_threshold=12
|
||
|
lbase_max_bytes=67108864
|
||
|
max_concurrent_compactions=1
|
||
|
max_manifest_file_size=134217728
|
||
|
max_open_files=1000
|
||
|
mem_table_size=4194304
|
||
|
mem_table_stop_writes_threshold=2
|
||
|
min_deletion_rate=0
|
||
|
merger=pebble.concatenate
|
||
|
read_compaction_rate=16000
|
||
|
read_sampling_multiplier=16
|
||
|
strict_wal_tail=true
|
||
|
table_cache_shards=8
|
||
|
table_property_collectors=[]
|
||
|
validate_on_ingest=false
|
||
|
wal_dir=
|
||
|
wal_bytes_per_sync=0
|
||
|
max_writer_concurrency=0
|
||
|
force_writer_parallelism=false
|
||
|
secondary_cache_size_bytes=0
|
||
|
create_on_shared=0
|
||
|
|
||
|
[Level "0"]
|
||
|
block_restart_interval=16
|
||
|
block_size=4096
|
||
|
block_size_threshold=90
|
||
|
compression=Snappy
|
||
|
filter_policy=none
|
||
|
filter_type=table
|
||
|
index_block_size=4096
|
||
|
target_file_size=2097152
|
||
|
`
|
||
|
|
||
|
var opts *Options
|
||
|
opts = opts.EnsureDefaults()
|
||
|
if v := opts.String(); expected != v {
|
||
|
t.Fatalf("expected\n%s\nbut found\n%s", expected, v)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func TestOptionsCheck(t *testing.T) {
|
||
|
var opts *Options
|
||
|
opts = opts.EnsureDefaults()
|
||
|
s := opts.String()
|
||
|
require.NoError(t, opts.Check(s))
|
||
|
require.Regexp(t, `invalid key=value syntax`, opts.Check("foo\n"))
|
||
|
|
||
|
tmp := *opts
|
||
|
tmp.Comparer = &Comparer{Name: "foo"}
|
||
|
require.Regexp(t, `comparer name from file.*!=.*`, tmp.Check(s))
|
||
|
|
||
|
tmp = *opts
|
||
|
tmp.Merger = &Merger{Name: "foo"}
|
||
|
require.Regexp(t, `merger name from file.*!=.*`, tmp.Check(s))
|
||
|
|
||
|
// RocksDB uses a similar (INI-style) syntax for the OPTIONS file, but
|
||
|
// different section names and keys.
|
||
|
s = `
|
||
|
[CFOptions "default"]
|
||
|
comparator=rocksdb-comparer
|
||
|
merge_operator=rocksdb-merger
|
||
|
`
|
||
|
tmp = *opts
|
||
|
tmp.Comparer = &Comparer{Name: "foo"}
|
||
|
require.Regexp(t, `comparer name from file.*!=.*`, tmp.Check(s))
|
||
|
|
||
|
tmp.Comparer = &Comparer{Name: "rocksdb-comparer"}
|
||
|
tmp.Merger = &Merger{Name: "foo"}
|
||
|
require.Regexp(t, `merger name from file.*!=.*`, tmp.Check(s))
|
||
|
|
||
|
tmp.Merger = &Merger{Name: "rocksdb-merger"}
|
||
|
require.NoError(t, tmp.Check(s))
|
||
|
|
||
|
// RocksDB allows the merge operator to be unspecified, in which case it
|
||
|
// shows up as "nullptr".
|
||
|
s = `
|
||
|
[CFOptions "default"]
|
||
|
merge_operator=nullptr
|
||
|
`
|
||
|
tmp = *opts
|
||
|
require.NoError(t, tmp.Check(s))
|
||
|
}
|
||
|
|
||
|
type testCleaner struct{}
|
||
|
|
||
|
func (testCleaner) Clean(fs vfs.FS, fileType base.FileType, path string) error {
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (testCleaner) String() string {
|
||
|
return "test-cleaner"
|
||
|
}
|
||
|
|
||
|
func TestOptionsParse(t *testing.T) {
|
||
|
testComparer := *DefaultComparer
|
||
|
testComparer.Name = "test-comparer"
|
||
|
testMerger := *DefaultMerger
|
||
|
testMerger.Name = "test-merger"
|
||
|
var newCacheSize int64
|
||
|
|
||
|
hooks := &ParseHooks{
|
||
|
NewCache: func(size int64) *Cache {
|
||
|
newCacheSize = size
|
||
|
return nil
|
||
|
},
|
||
|
NewCleaner: func(name string) (Cleaner, error) {
|
||
|
if name == (testCleaner{}).String() {
|
||
|
return testCleaner{}, nil
|
||
|
}
|
||
|
return nil, errors.Errorf("unknown cleaner: %q", name)
|
||
|
},
|
||
|
NewComparer: func(name string) (*Comparer, error) {
|
||
|
if name == testComparer.Name {
|
||
|
return &testComparer, nil
|
||
|
}
|
||
|
return nil, errors.Errorf("unknown comparer: %q", name)
|
||
|
},
|
||
|
NewMerger: func(name string) (*Merger, error) {
|
||
|
if name == testMerger.Name {
|
||
|
return &testMerger, nil
|
||
|
}
|
||
|
return nil, errors.Errorf("unknown merger: %q", name)
|
||
|
},
|
||
|
}
|
||
|
|
||
|
testCases := []struct {
|
||
|
cleaner Cleaner
|
||
|
comparer *Comparer
|
||
|
merger *Merger
|
||
|
}{
|
||
|
{testCleaner{}, nil, nil},
|
||
|
{nil, &testComparer, nil},
|
||
|
{nil, nil, &testMerger},
|
||
|
}
|
||
|
for _, c := range testCases {
|
||
|
t.Run("", func(t *testing.T) {
|
||
|
var opts Options
|
||
|
opts.Comparer = c.comparer
|
||
|
opts.Merger = c.merger
|
||
|
opts.WALDir = "wal"
|
||
|
opts.Levels = make([]LevelOptions, 3)
|
||
|
opts.Levels[0].BlockSize = 1024
|
||
|
opts.Levels[1].BlockSize = 2048
|
||
|
opts.Levels[2].BlockSize = 4096
|
||
|
opts.Experimental.CompactionDebtConcurrency = 100
|
||
|
opts.FlushDelayDeleteRange = 10 * time.Second
|
||
|
opts.FlushDelayRangeKey = 11 * time.Second
|
||
|
opts.Experimental.LevelMultiplier = 5
|
||
|
opts.TargetByteDeletionRate = 200
|
||
|
opts.Experimental.ReadCompactionRate = 300
|
||
|
opts.Experimental.ReadSamplingMultiplier = 400
|
||
|
opts.Experimental.TableCacheShards = 500
|
||
|
opts.Experimental.MaxWriterConcurrency = 1
|
||
|
opts.Experimental.ForceWriterParallelism = true
|
||
|
opts.Experimental.SecondaryCacheSizeBytes = 1024
|
||
|
opts.EnsureDefaults()
|
||
|
str := opts.String()
|
||
|
|
||
|
newCacheSize = 0
|
||
|
var parsedOptions Options
|
||
|
require.NoError(t, parsedOptions.Parse(str, hooks))
|
||
|
parsedStr := parsedOptions.String()
|
||
|
if str != parsedStr {
|
||
|
t.Fatalf("expected\n%s\nbut found\n%s", str, parsedStr)
|
||
|
}
|
||
|
require.Nil(t, parsedOptions.Cache)
|
||
|
require.NotEqual(t, newCacheSize, 0)
|
||
|
})
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func TestOptionsValidate(t *testing.T) {
|
||
|
testCases := []struct {
|
||
|
options string
|
||
|
expected string
|
||
|
}{
|
||
|
{``, ``},
|
||
|
{`
|
||
|
[Options]
|
||
|
l0_compaction_concurrency=0
|
||
|
`,
|
||
|
`L0CompactionConcurrency \(0\) must be >= 1`,
|
||
|
},
|
||
|
{`
|
||
|
[Options]
|
||
|
l0_compaction_threshold=2
|
||
|
l0_stop_writes_threshold=1
|
||
|
`,
|
||
|
`L0StopWritesThreshold .* must be >= L0CompactionThreshold .*`,
|
||
|
},
|
||
|
{`
|
||
|
[Options]
|
||
|
mem_table_size=4294967296
|
||
|
`,
|
||
|
`MemTableSize \(4\.0GB\) must be < [2|4]\.0GB`,
|
||
|
},
|
||
|
{`
|
||
|
[Options]
|
||
|
mem_table_stop_writes_threshold=1
|
||
|
`,
|
||
|
`MemTableStopWritesThreshold .* must be >= 2`,
|
||
|
},
|
||
|
}
|
||
|
|
||
|
for _, c := range testCases {
|
||
|
t.Run("", func(t *testing.T) {
|
||
|
var opts Options
|
||
|
opts.EnsureDefaults()
|
||
|
require.NoError(t, opts.Parse(c.options, nil))
|
||
|
err := opts.Validate()
|
||
|
if c.expected == "" {
|
||
|
require.NoError(t, err)
|
||
|
} else {
|
||
|
require.Error(t, err)
|
||
|
require.Regexp(t, c.expected, err.Error())
|
||
|
}
|
||
|
})
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// This test isn't being done in TestOptionsValidate
|
||
|
// cause it doesn't support setting pointers.
|
||
|
func TestOptionsValidateCache(t *testing.T) {
|
||
|
var opts Options
|
||
|
opts.EnsureDefaults()
|
||
|
opts.Cache = NewCache(8 << 20)
|
||
|
defer opts.Cache.Unref()
|
||
|
opts.TableCache = NewTableCache(NewCache(8<<20), 10, 1)
|
||
|
defer opts.TableCache.cache.Unref()
|
||
|
defer opts.TableCache.Unref()
|
||
|
|
||
|
err := opts.Validate()
|
||
|
require.Error(t, err)
|
||
|
if fmt.Sprint(err) != "underlying cache in the TableCache and the Cache dont match" {
|
||
|
t.Errorf("Unexpected error message")
|
||
|
}
|
||
|
}
|