erigon-pulse/ethdb/snapshotdb/kv_snapshot_property_test.go
2021-07-29 18:53:13 +07:00

467 lines
12 KiB
Go

package snapshotdb
import (
"context"
"testing"
"github.com/ledgerwatch/erigon-lib/kv"
"github.com/ledgerwatch/erigon-lib/kv/memdb"
"github.com/ledgerwatch/erigon/common"
"github.com/stretchr/testify/require"
"pgregory.net/rapid"
)
func TestGetAndPut(t *testing.T) {
t.Skip("remove when it become stable for 200 rounds")
rapid.Check(t, rapid.Run(&getPutkvMachine{}))
}
type getPutkvMachine struct {
bucket string
snKV kv.RwDB
modelKV kv.RwDB
snapshotKeys [][20]byte
newKeys [][20]byte
allKeys [][20]byte
snTX kv.RwTx
modelTX kv.RwTx
}
func (m *getPutkvMachine) Init(t *rapid.T) {
m.bucket = kv.PlainState
m.snKV = memdb.New()
m.modelKV = memdb.New()
m.snapshotKeys = rapid.SliceOf(rapid.ArrayOf(20, rapid.Byte())).Filter(func(_v [][20]byte) bool {
return len(_v) > 0
}).Draw(t, "generate keys").([][20]byte)
m.newKeys = rapid.SliceOf(rapid.ArrayOf(20, rapid.Byte())).Filter(func(_v [][20]byte) bool {
return len(_v) > 0
}).Draw(t, "generate new keys").([][20]byte)
notExistingKeys := rapid.SliceOf(rapid.ArrayOf(20, rapid.Byte())).Filter(func(_v [][20]byte) bool {
return len(_v) > 0
}).Draw(t, "generate not excisting keys").([][20]byte)
m.allKeys = append(m.snapshotKeys, notExistingKeys...)
txSn, err := m.snKV.BeginRw(context.Background())
require.NoError(t, err)
defer txSn.Rollback()
txModel, err := m.modelKV.BeginRw(context.Background())
require.NoError(t, err)
defer txModel.Rollback()
for _, key := range m.snapshotKeys {
innerErr := txSn.Put(m.bucket, key[:], []byte("sn_"+common.Bytes2Hex(key[:])))
require.NoError(t, innerErr)
innerErr = txModel.Put(m.bucket, key[:], []byte("sn_"+common.Bytes2Hex(key[:])))
require.NoError(t, innerErr)
}
//save snapshot and wrap new write db
err = txSn.Commit()
require.NoError(t, err)
err = txModel.Commit()
require.NoError(t, err)
m.snKV = NewSnapshotKV().StateSnapshot(m.snKV).DB(memdb.New()).Open()
}
func (m *getPutkvMachine) Cleanup() {
if m.snTX != nil {
m.snTX.Rollback()
}
if m.modelTX != nil {
m.modelTX.Rollback()
}
m.snKV.Close()
m.modelKV.Close()
}
func (m *getPutkvMachine) Check(t *rapid.T) {
}
func (m *getPutkvMachine) Get(t *rapid.T) {
if m.snTX == nil && m.modelTX == nil {
return
}
key := rapid.SampledFrom(m.allKeys).Draw(t, "get a key").([20]byte)
var (
v1, v2 []byte
err1, err2 error
)
v1, err1 = m.snTX.GetOne(m.bucket, key[:])
v2, err2 = m.modelTX.GetOne(m.bucket, key[:])
require.Equal(t, err1, err2)
require.Equal(t, v1, v2)
}
func (m *getPutkvMachine) Put(t *rapid.T) {
if len(m.newKeys) == 0 {
return
}
if m.snTX == nil && m.modelTX == nil {
return
}
key := rapid.SampledFrom(m.newKeys).Draw(t, "put a key").([20]byte)
m.allKeys = append(m.allKeys, key)
for i, v := range m.newKeys {
if v == key {
m.newKeys = append(m.newKeys[:i], m.newKeys[i+1:]...)
}
}
err := m.snTX.Put(m.bucket, key[:], []byte("put"+common.Bytes2Hex(key[:])))
require.NoError(t, err)
err = m.modelTX.Put(m.bucket, key[:], []byte("put"+common.Bytes2Hex(key[:])))
require.NoError(t, err)
}
func (m *getPutkvMachine) Delete(t *rapid.T) {
if m.snTX == nil && m.modelTX == nil {
return
}
key := rapid.SampledFrom(m.allKeys).Draw(t, "delete a key").([20]byte)
err := m.snTX.Delete(m.bucket, key[:], nil)
require.NoError(t, err)
err = m.modelTX.Put(m.bucket, key[:], nil)
require.NoError(t, err)
}
func (m *getPutkvMachine) Begin(t *rapid.T) {
if m.modelTX != nil && m.snTX != nil {
return
}
mtx, err := m.modelKV.BeginRw(context.Background()) //nolint
require.NoError(t, err)
sntx, err := m.snKV.BeginRw(context.Background()) //nolint
require.NoError(t, err)
m.modelTX = mtx
m.snTX = sntx
}
func (m *getPutkvMachine) Rollback(t *rapid.T) {
if m.modelTX == nil && m.snTX == nil {
return
}
m.snTX.Rollback()
m.modelTX.Rollback()
m.snTX = nil
m.modelTX = nil
}
func (m *getPutkvMachine) Commit(t *rapid.T) {
if m.modelTX == nil && m.snTX == nil {
return
}
err := m.modelTX.Commit()
require.NoError(t, err)
err = m.snTX.Commit()
require.NoError(t, err)
m.snTX = nil
m.modelTX = nil
}
type getKVMachine struct {
bucket string
snKV kv.RwDB
modelKV kv.RwDB
overWriteKeys [][20]byte
snKeys [][20]byte
newKeys [][20]byte
allKeys [][20]byte
}
func (m *getKVMachine) Init(t *rapid.T) {
m.bucket = kv.PlainState
m.snKV = memdb.New()
m.modelKV = memdb.New()
m.snKeys = rapid.SliceOf(rapid.ArrayOf(20, rapid.Byte())).Filter(func(_v [][20]byte) bool {
return len(_v) > 0
}).Draw(t, "generate keys").([][20]byte)
m.overWriteKeys = rapid.SliceOf(rapid.SampledFrom(m.snKeys)).Draw(t, "get snKeys").([][20]byte)
m.newKeys = rapid.SliceOf(rapid.ArrayOf(20, rapid.Byte())).Draw(t, "generate new keys").([][20]byte)
m.allKeys = append(m.snKeys, m.overWriteKeys...)
m.allKeys = append(m.allKeys, m.newKeys...)
notExistingKeys := rapid.SliceOf(rapid.ArrayOf(20, rapid.Byte())).Draw(t, "generate not excisting keys").([][20]byte)
m.allKeys = append(m.allKeys, notExistingKeys...)
txSn, err := m.snKV.BeginRw(context.Background())
require.NoError(t, err)
defer txSn.Rollback()
txModel, err := m.modelKV.BeginRw(context.Background())
require.NoError(t, err)
defer txModel.Rollback()
for _, key := range m.snKeys {
innerErr := txSn.Put(m.bucket, key[:], []byte("sn_"+common.Bytes2Hex(key[:])))
require.NoError(t, innerErr)
innerErr = txModel.Put(m.bucket, key[:], []byte("sn_"+common.Bytes2Hex(key[:])))
require.NoError(t, innerErr)
}
//save snapshot and wrap new write db
err = txSn.Commit()
require.NoError(t, err)
m.snKV = NewSnapshotKV().StateSnapshot(m.snKV).DB(memdb.New()).Open()
txSn, err = m.snKV.BeginRw(context.Background())
require.NoError(t, err)
defer txSn.Rollback()
for _, key := range m.overWriteKeys {
innerErr := txSn.Put(m.bucket, key[:], []byte("overwrite_"+common.Bytes2Hex(key[:])))
require.NoError(t, innerErr)
innerErr = txModel.Put(m.bucket, key[:], []byte("overwrite_"+common.Bytes2Hex(key[:])))
require.NoError(t, innerErr)
}
for _, key := range m.newKeys {
innerErr := txSn.Put(m.bucket, key[:], []byte("new_"+common.Bytes2Hex(key[:])))
require.NoError(t, innerErr)
innerErr = txModel.Put(m.bucket, key[:], []byte("new_"+common.Bytes2Hex(key[:])))
require.NoError(t, innerErr)
}
err = txSn.Commit()
require.NoError(t, err)
err = txModel.Commit()
require.NoError(t, err)
}
func (m *getKVMachine) Cleanup() {
m.snKV.Close()
m.modelKV.Close()
}
func (m *getKVMachine) Check(t *rapid.T) {
}
func (m *getKVMachine) Get(t *rapid.T) {
key := rapid.SampledFrom(m.allKeys).Draw(t, "get a key").([20]byte)
var (
v1, v2 []byte
err1, err2 error
)
err := m.snKV.View(context.Background(), func(tx kv.Tx) error {
v1, err1 = tx.GetOne(m.bucket, key[:])
return nil
})
require.NoError(t, err)
err = m.modelKV.View(context.Background(), func(tx kv.Tx) error {
v2, err2 = tx.GetOne(m.bucket, key[:])
return nil
})
require.NoError(t, err)
require.Equal(t, err1, err2)
require.Equal(t, v1, v2)
}
func TestGet(t *testing.T) {
t.Skip("remove when it become stable for 200 rounds")
rapid.Check(t, rapid.Run(&getKVMachine{}))
}
func TestCursorWithTX(t *testing.T) {
t.Skip("remove when it become stable for 200 rounds")
rapid.Check(t, rapid.Run(&cursorKVMachine{}))
}
type cursorKVMachine struct {
bucket string
snKV kv.RwDB
modelKV kv.RwDB
snTX kv.RwTx
modelTX kv.RwTx
snCursor kv.RwCursor
modelCursor kv.RwCursor
snapshotKeys [][20]byte
newKeys [][20]byte
allKeys [][20]byte
}
func (m *cursorKVMachine) Init(t *rapid.T) {
m.bucket = kv.PlainState
m.snKV = memdb.New()
m.modelKV = memdb.New()
m.snapshotKeys = rapid.SliceOf(rapid.ArrayOf(20, rapid.Byte())).Filter(func(_v [][20]byte) bool {
return len(_v) > 0
}).Draw(t, "generate keys").([][20]byte)
m.newKeys = rapid.SliceOf(rapid.ArrayOf(20, rapid.Byte())).Filter(func(_v [][20]byte) bool {
return len(_v) > 0
}).Draw(t, "generate new keys").([][20]byte)
notExistingKeys := rapid.SliceOf(rapid.ArrayOf(20, rapid.Byte())).Filter(func(_v [][20]byte) bool {
return len(_v) > 0
}).Draw(t, "generate not excisting keys").([][20]byte)
m.allKeys = append(m.snapshotKeys, notExistingKeys...)
defer func() {
m.snTX = nil
m.modelTX = nil
}()
txSn, err := m.snKV.BeginRw(context.Background())
require.NoError(t, err)
defer txSn.Rollback()
txModel, err := m.modelKV.BeginRw(context.Background())
require.NoError(t, err)
defer txModel.Rollback()
for _, key := range m.snapshotKeys {
innerErr := txSn.Put(m.bucket, key[:], []byte("sn_"+common.Bytes2Hex(key[:])))
require.NoError(t, innerErr)
innerErr = txModel.Put(m.bucket, key[:], []byte("sn_"+common.Bytes2Hex(key[:])))
require.NoError(t, innerErr)
}
//save snapshot and wrap new write db
err = txSn.Commit()
require.NoError(t, err)
err = txModel.Commit()
require.NoError(t, err)
m.snKV = NewSnapshotKV().StateSnapshot(m.snKV).DB(memdb.New()).Open()
}
func (m *cursorKVMachine) Check(t *rapid.T) {
}
func (m *cursorKVMachine) Cleanup() {
if m.snTX != nil {
m.snTX.Rollback()
}
if m.modelTX != nil {
m.modelTX.Rollback()
}
m.snKV.Close()
m.snKV = nil
m.modelKV.Close()
m.modelKV = nil
}
func (m *cursorKVMachine) Begin(t *rapid.T) {
if m.modelTX != nil && m.snTX != nil {
return
}
mtx, err := m.modelKV.BeginRw(context.Background()) //nolint
require.NoError(t, err)
sntx, err := m.snKV.BeginRw(context.Background()) //nolint
require.NoError(t, err)
m.modelTX = mtx
m.snTX = sntx
}
func (m *cursorKVMachine) Rollback(t *rapid.T) {
if m.modelTX == nil && m.snTX == nil {
return
}
m.snTX.Rollback()
m.modelTX.Rollback()
m.snTX = nil
m.modelTX = nil
m.snCursor = nil
m.modelCursor = nil
}
func (m *cursorKVMachine) Commit(t *rapid.T) {
if m.modelTX == nil && m.snTX == nil {
return
}
err := m.modelTX.Commit()
require.NoError(t, err)
err = m.snTX.Commit()
require.NoError(t, err)
m.snTX = nil
m.modelTX = nil
m.snCursor = nil
m.modelCursor = nil
}
func (m *cursorKVMachine) Cursor(t *rapid.T) {
if m.modelTX == nil && m.snTX == nil {
return
}
if m.modelCursor != nil && m.snCursor != nil {
return
}
var err error
m.modelCursor, err = m.modelTX.RwCursor(m.bucket)
require.NoError(t, err)
m.snCursor, err = m.snTX.RwCursor(m.bucket)
require.NoError(t, err)
}
func (m *cursorKVMachine) CloseCursor(t *rapid.T) {
if m.modelTX == nil && m.snTX == nil {
return
}
if m.modelCursor == nil && m.snCursor == nil {
return
}
m.modelCursor.Close()
m.snCursor.Close()
m.modelCursor = nil
m.snCursor = nil
}
func (m *cursorKVMachine) First(t *rapid.T) {
if m.modelCursor == nil && m.snCursor == nil {
return
}
k1, v1, err1 := m.modelCursor.First()
k2, v2, err2 := m.snCursor.First()
require.Equal(t, k1, k2)
require.Equal(t, v1, v2)
require.Equal(t, err1, err2)
}
func (m *cursorKVMachine) Last(t *rapid.T) {
if m.modelCursor == nil && m.snCursor == nil {
return
}
k1, v1, err1 := m.modelCursor.Last()
k2, v2, err2 := m.snCursor.Last()
require.Equal(t, k1, k2)
require.Equal(t, v1, v2)
require.Equal(t, err1, err2)
}
func (m *cursorKVMachine) Seek(t *rapid.T) {
if m.modelCursor == nil && m.snCursor == nil {
return
}
key := rapid.SampledFrom(m.allKeys).Draw(t, "get random key").([20]byte)
k1, v1, err1 := m.modelCursor.Seek(key[:])
k2, v2, err2 := m.snCursor.Seek(key[:])
require.Equal(t, k1, k2)
require.Equal(t, v1, v2)
require.Equal(t, err1, err2)
}
func (m *cursorKVMachine) SeekExact(t *rapid.T) {
if m.modelCursor == nil && m.snCursor == nil {
return
}
key := rapid.SampledFrom(m.allKeys).Draw(t, "get random key").([20]byte)
k1, v1, err1 := m.modelCursor.SeekExact(key[:])
k2, v2, err2 := m.snCursor.SeekExact(key[:])
require.Equal(t, k1, k2)
require.Equal(t, v1, v2)
require.Equal(t, err1, err2)
}
func (m *cursorKVMachine) Next(t *rapid.T) {
if m.modelCursor == nil && m.snCursor == nil {
return
}
k1, v1, err1 := m.modelCursor.Next()
k2, v2, err2 := m.snCursor.Next()
require.Equal(t, k1, k2)
require.Equal(t, v1, v2)
require.Equal(t, err1, err2)
}