Skip to content

Commit

Permalink
engine/metamorphic: Add engine restarts, compare across runs`
Browse files Browse the repository at this point in the history
This PR builds on top of cockroachdb#44458 (all commits before the last one
are from that PR). It adds two things: one, it ensures that whenever
successive engine types are tested, it does a comparison for matching
outputs. This will ensure CI will fail if any change down the line
causes an observed difference in behaviour between rocksdb and pebble.

It also adds more MVCC operations that would be useful to test, such
as MVCCFindSplitKey, MVCCDeleteRange, MVCCClearTimeRange,
MVCCConditionalPut, etc.

Furthermore, it adds a new kind of operation: restart. Restarts are
special in that they're added after the specified n number of runs
have happened; so a test run with 3 specified engine types
(so 2 restarts), and n = 10000 will have 3 * 10000 = 30000 operations.
A restart closes all open objects, then closes the engine, and starts
up the next engine in the specified engine sequence. This, combined
with the aforementioned checking functionality across different
engine runs, lets us test for bidirectional compatibility across
different engines.

Fixes cockroachdb#43762 .

Release note: None.
  • Loading branch information
itsbilal committed Feb 7, 2020
1 parent 51e473d commit 07e243b
Show file tree
Hide file tree
Showing 4 changed files with 484 additions and 124 deletions.
103 changes: 99 additions & 4 deletions pkg/storage/engine/metamorphic/generator.go
Original file line number Diff line number Diff line change
Expand Up @@ -19,13 +19,62 @@ import (
"strings"
"testing"

"github.com/cockroachdb/cockroach/pkg/base"
"github.com/cockroachdb/cockroach/pkg/roachpb"
"github.com/cockroachdb/cockroach/pkg/settings/cluster"
"github.com/cockroachdb/cockroach/pkg/storage/engine"
"github.com/cockroachdb/cockroach/pkg/util/hlc"
"github.com/cockroachdb/pebble"
)

const zipfMax uint64 = 100000

func makeStorageConfig(path string) base.StorageConfig {
return base.StorageConfig{
Attrs: roachpb.Attributes{},
Dir: path,
MustExist: false,
MaxSize: 0,
Settings: cluster.MakeTestingClusterSettings(),
UseFileRegistry: false,
ExtraOptions: nil,
}
}

func createTestRocksDBEngine(path string) (engine.Engine, error) {
cache := engine.NewRocksDBCache(1<<20)
defer cache.Release()
cfg := engine.RocksDBConfig{
StorageConfig: makeStorageConfig(path),
ReadOnly: false,
}

return engine.NewRocksDB(cfg, cache)
}

func createTestPebbleEngine(path string) (engine.Engine, error) {
pebbleConfig := engine.PebbleConfig{
StorageConfig: makeStorageConfig(path),
Opts: engine.DefaultPebbleOptions(),
}
pebbleConfig.Opts.Cache = pebble.NewCache(1 << 20)

return engine.NewPebble(context.Background(), pebbleConfig)
}

type engineImpl struct{
name string
create func(path string) (engine.Engine, error)
}
var _ fmt.Stringer = &engineImpl{}

func (e *engineImpl) String() string {
return e.name
}

var engineImplRocksDB = engineImpl{"rocksdb", createTestRocksDBEngine}
var engineImplPebble = engineImpl{"pebble", createTestPebbleEngine}

// Object to store info corresponding to one metamorphic test run. Responsible
// for generating and executing operations.
type metaTestRunner struct {
Expand All @@ -34,6 +83,10 @@ type metaTestRunner struct {
t *testing.T
rng *rand.Rand
seed int64
path string
engineImpls []engineImpl
curEngine int
numRestarts int
engine engine.Engine
tsGenerator tsGenerator
managers map[operandType]operandManager
Expand All @@ -46,6 +99,14 @@ func (m *metaTestRunner) init() {
// test runs should guarantee the same operations being generated.
m.rng = rand.New(rand.NewSource(m.seed))
m.tsGenerator.init(m.rng)
m.numRestarts = len(m.engineImpls) - 1
m.curEngine = 0

var err error
m.engine, err = m.engineImpls[0].create(m.path)
if err != nil {
m.t.Fatal(err)
}

m.managers = map[operandType]operandManager{
operandTransaction: &txnManager{
Expand All @@ -57,7 +118,7 @@ func (m *metaTestRunner) init() {
},
operandReadWriter: &readWriterManager{
rng: m.rng,
eng: m.engine,
m: m,
batchToIDMap: make(map[engine.Batch]int),
},
operandMVCCKey: &keyManager{
Expand Down Expand Up @@ -89,6 +150,10 @@ func (m *metaTestRunner) init() {
// Run this function in a defer to ensure any Fatals on m.t do not cause panics
// due to leaked iterators.
func (m *metaTestRunner) closeAll() {
if m.engine == nil {
// Engine already closed; possibly running in a defer after a panic.
return
}
// Close all open objects. This should let the engine close cleanly.
closingOrder := []operandType{
operandIterator,
Expand All @@ -98,17 +163,47 @@ func (m *metaTestRunner) closeAll() {
for _, operandType := range closingOrder {
m.managers[operandType].closeAll()
}
m.engine.Close()
m.engine = nil
}

// generateAndRun generates n operations using a TPCC-style deck shuffle with
// weighted probabilities of each operation appearing.
func (m *metaTestRunner) generateAndRun(n int) {
deck := newDeck(m.rng, m.weights...)

for i := 0; i < n; i++ {
op := &operations[deck.Int()]
for numRestart := 0; numRestart <= m.numRestarts; numRestart++ {
if numRestart > 0 {
// Insert and run a restart operation.
m.runOp(opRun{
op: m.nameToOp["restart"],
args: nil,
})
}
for i := 0; i < n; i++ {
op := &operations[deck.Int()]

m.resolveAndRunOp(op)
}
}
}

// Closes the current engine and starts another one up, with the same path.
func (m *metaTestRunner) restart() {
m.closeAll()
m.curEngine++
if m.curEngine > m.numRestarts {
// If we're restarting more times than the number of engine implementations
// specified, just restart with the last engine type again. This is useful
// for when a check file with multiple restarts is passed in for a run with
// only one engine type specified.
m.curEngine = m.numRestarts
}

m.resolveAndRunOp(op)
var err error
m.engine, err = m.engineImpls[m.curEngine].create(m.path)
if err != nil {
m.t.Fatal(err)
}
}

Expand Down
Loading

0 comments on commit 07e243b

Please sign in to comment.