mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-14 23:08:20 +00:00
467 lines
12 KiB
Go
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)
|
|
}
|