erigon-pulse/kv/memdb/memory_mutation_test.go

686 lines
20 KiB
Go

/*
Copyright 2022 Erigon 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 memdb
import (
"context"
"testing"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
"github.com/ledgerwatch/erigon-lib/kv"
)
func initializeDbNonDupSort(rwTx kv.RwTx) {
rwTx.Put(kv.HashedAccounts, []byte("AAAA"), []byte("value"))
rwTx.Put(kv.HashedAccounts, []byte("CAAA"), []byte("value1"))
rwTx.Put(kv.HashedAccounts, []byte("CBAA"), []byte("value2"))
rwTx.Put(kv.HashedAccounts, []byte("CCAA"), []byte("value3"))
}
func TestPutAppendHas(t *testing.T) {
rwTx, err := New().BeginRw(context.Background())
require.Nil(t, err)
initializeDbNonDupSort(rwTx)
batch := NewMemoryBatch(rwTx, "")
require.NoError(t, batch.Append(kv.HashedAccounts, []byte("AAAA"), []byte("value1.5")))
require.Error(t, batch.Append(kv.HashedAccounts, []byte("AAAA"), []byte("value1.3")))
require.NoError(t, batch.Put(kv.HashedAccounts, []byte("AAAA"), []byte("value1.3")))
require.NoError(t, batch.Append(kv.HashedAccounts, []byte("CBAA"), []byte("value3.5")))
require.Error(t, batch.Append(kv.HashedAccounts, []byte("CBAA"), []byte("value3.1")))
require.NoError(t, batch.AppendDup(kv.HashedAccounts, []byte("CBAA"), []byte("value3.1")))
require.Nil(t, batch.Flush(rwTx))
exist, err := batch.Has(kv.HashedAccounts, []byte("AAAA"))
require.Nil(t, err)
require.Equal(t, exist, true)
val, err := batch.GetOne(kv.HashedAccounts, []byte("AAAA"))
require.Nil(t, err)
require.Equal(t, val, []byte("value1.3"))
exist, err = batch.Has(kv.HashedAccounts, []byte("KKKK"))
require.Nil(t, err)
require.Equal(t, exist, false)
}
func TestLastMiningDB(t *testing.T) {
rwTx, err := New().BeginRw(context.Background())
require.NoError(t, err)
initializeDbNonDupSort(rwTx)
batch := NewMemoryBatch(rwTx, "")
batch.Put(kv.HashedAccounts, []byte("BAAA"), []byte("value4"))
batch.Put(kv.HashedAccounts, []byte("BCAA"), []byte("value5"))
cursor, err := batch.Cursor(kv.HashedAccounts)
require.NoError(t, err)
key, value, err := cursor.Last()
require.NoError(t, err)
require.Equal(t, key, []byte("CCAA"))
require.Equal(t, value, []byte("value3"))
key, value, err = cursor.Next()
require.NoError(t, err)
require.Equal(t, key, []byte(nil))
require.Equal(t, value, []byte(nil))
}
func TestLastMiningMem(t *testing.T) {
rwTx, err := New().BeginRw(context.Background())
require.NoError(t, err)
initializeDbNonDupSort(rwTx)
batch := NewMemoryBatch(rwTx, "")
batch.Put(kv.HashedAccounts, []byte("BAAA"), []byte("value4"))
batch.Put(kv.HashedAccounts, []byte("DCAA"), []byte("value5"))
cursor, err := batch.Cursor(kv.HashedAccounts)
require.NoError(t, err)
key, value, err := cursor.Last()
require.NoError(t, err)
require.Equal(t, key, []byte("DCAA"))
require.Equal(t, value, []byte("value5"))
key, value, err = cursor.Next()
require.NoError(t, err)
require.Equal(t, key, []byte(nil))
require.Equal(t, value, []byte(nil))
}
func TestDeleteMining(t *testing.T) {
rwTx, err := New().BeginRw(context.Background())
require.NoError(t, err)
initializeDbNonDupSort(rwTx)
batch := NewMemoryBatch(rwTx, "")
batch.Put(kv.HashedAccounts, []byte("BAAA"), []byte("value4"))
batch.Put(kv.HashedAccounts, []byte("DCAA"), []byte("value5"))
batch.Put(kv.HashedAccounts, []byte("FCAA"), []byte("value5"))
batch.Delete(kv.HashedAccounts, []byte("BAAA"))
batch.Delete(kv.HashedAccounts, []byte("CBAA"))
cursor, err := batch.Cursor(kv.HashedAccounts)
require.NoError(t, err)
key, value, err := cursor.SeekExact([]byte("BAAA"))
require.NoError(t, err)
require.Equal(t, key, []byte(nil))
require.Equal(t, value, []byte(nil))
key, value, err = cursor.SeekExact([]byte("CBAA"))
require.NoError(t, err)
require.Equal(t, key, []byte(nil))
require.Equal(t, value, []byte(nil))
}
func TestFlush(t *testing.T) {
rwTx, err := New().BeginRw(context.Background())
require.NoError(t, err)
initializeDbNonDupSort(rwTx)
batch := NewMemoryBatch(rwTx, "")
batch.Put(kv.HashedAccounts, []byte("BAAA"), []byte("value4"))
batch.Put(kv.HashedAccounts, []byte("AAAA"), []byte("value5"))
batch.Put(kv.HashedAccounts, []byte("FCAA"), []byte("value5"))
require.NoError(t, batch.Flush(rwTx))
value, err := rwTx.GetOne(kv.HashedAccounts, []byte("BAAA"))
require.NoError(t, err)
require.Equal(t, value, []byte("value4"))
value, err = rwTx.GetOne(kv.HashedAccounts, []byte("AAAA"))
require.NoError(t, err)
require.Equal(t, value, []byte("value5"))
}
func TestForEach(t *testing.T) {
rwTx, err := New().BeginRw(context.Background())
require.NoError(t, err)
initializeDbNonDupSort(rwTx)
batch := NewMemoryBatch(rwTx, "")
batch.Put(kv.HashedAccounts, []byte("FCAA"), []byte("value5"))
require.NoError(t, batch.Flush(rwTx))
var keys []string
var values []string
err = batch.ForEach(kv.HashedAccounts, []byte("XYAZ"), func(k, v []byte) error {
keys = append(keys, string(k))
values = append(values, string(v))
return nil
})
require.Nil(t, err)
require.Nil(t, keys)
require.Nil(t, values)
err = batch.ForEach(kv.HashedAccounts, []byte("CC"), func(k, v []byte) error {
keys = append(keys, string(k))
values = append(values, string(v))
return nil
})
require.Nil(t, err)
require.Equal(t, []string{"CCAA", "FCAA"}, keys)
require.Equal(t, []string{"value3", "value5"}, values)
var keys1 []string
var values1 []string
err = batch.ForEach(kv.HashedAccounts, []byte("A"), func(k, v []byte) error {
keys1 = append(keys1, string(k))
values1 = append(values1, string(v))
return nil
})
require.Nil(t, err)
require.Equal(t, []string{"AAAA", "CAAA", "CBAA", "CCAA", "FCAA"}, keys1)
require.Equal(t, []string{"value", "value1", "value2", "value3", "value5"}, values1)
}
func TestForPrefix(t *testing.T) {
rwTx, err := New().BeginRw(context.Background())
require.NoError(t, err)
initializeDbNonDupSort(rwTx)
batch := NewMemoryBatch(rwTx, "")
var keys1 []string
var values1 []string
err = batch.ForPrefix(kv.HashedAccounts, []byte("AB"), func(k, v []byte) error {
keys1 = append(keys1, string(k))
values1 = append(values1, string(v))
return nil
})
require.Nil(t, err)
require.Nil(t, keys1)
require.Nil(t, values1)
err = batch.ForPrefix(kv.HashedAccounts, []byte("AAAA"), func(k, v []byte) error {
keys1 = append(keys1, string(k))
values1 = append(values1, string(v))
return nil
})
require.Nil(t, err)
require.Equal(t, []string{"AAAA"}, keys1)
require.Equal(t, []string{"value"}, values1)
var keys []string
var values []string
err = batch.ForPrefix(kv.HashedAccounts, []byte("C"), func(k, v []byte) error {
keys = append(keys, string(k))
values = append(values, string(v))
return nil
})
require.Nil(t, err)
require.Equal(t, []string{"CAAA", "CBAA", "CCAA"}, keys)
require.Equal(t, []string{"value1", "value2", "value3"}, values)
}
func TestForAmount(t *testing.T) {
rwTx, err := New().BeginRw(context.Background())
require.NoError(t, err)
initializeDbNonDupSort(rwTx)
batch := NewMemoryBatch(rwTx, "")
defer batch.Close()
var keys []string
var values []string
err = batch.ForAmount(kv.HashedAccounts, []byte("C"), uint32(3), func(k, v []byte) error {
keys = append(keys, string(k))
values = append(values, string(v))
return nil
})
require.Nil(t, err)
require.Equal(t, []string{"CAAA", "CBAA", "CCAA"}, keys)
require.Equal(t, []string{"value1", "value2", "value3"}, values)
var keys1 []string
var values1 []string
err = batch.ForAmount(kv.HashedAccounts, []byte("C"), uint32(10), func(k, v []byte) error {
keys1 = append(keys1, string(k))
values1 = append(values1, string(v))
return nil
})
require.Nil(t, err)
require.Equal(t, []string{"CAAA", "CBAA", "CCAA"}, keys1)
require.Equal(t, []string{"value1", "value2", "value3"}, values1)
}
func TestGetOneAfterClearBucket(t *testing.T) {
rwTx, err := New().BeginRw(context.Background())
require.NoError(t, err)
initializeDbNonDupSort(rwTx)
batch := NewMemoryBatch(rwTx, "")
defer batch.Close()
err = batch.ClearBucket(kv.HashedAccounts)
require.Nil(t, err)
cond := batch.isTableCleared(kv.HashedAccounts)
require.True(t, cond)
val, err := batch.GetOne(kv.HashedAccounts, []byte("A"))
require.Nil(t, err)
require.Nil(t, val)
val, err = batch.GetOne(kv.HashedAccounts, []byte("AAAA"))
require.Nil(t, err)
require.Nil(t, val)
}
func TestSeekExactAfterClearBucket(t *testing.T) {
rwTx, err := New().BeginRw(context.Background())
require.NoError(t, err)
initializeDbNonDupSort(rwTx)
batch := NewMemoryBatch(rwTx, "")
defer batch.Close()
err = batch.ClearBucket(kv.HashedAccounts)
require.Nil(t, err)
cond := batch.isTableCleared(kv.HashedAccounts)
require.True(t, cond)
cursor, err := batch.RwCursor(kv.HashedAccounts)
require.NoError(t, err)
key, val, err := cursor.SeekExact([]byte("AAAA"))
require.Nil(t, err)
assert.Nil(t, key)
assert.Nil(t, val)
err = cursor.Put([]byte("AAAA"), []byte("valueX"))
require.Nil(t, err)
key, val, err = cursor.SeekExact([]byte("AAAA"))
require.Nil(t, err)
assert.Equal(t, []byte("AAAA"), key)
assert.Equal(t, []byte("valueX"), val)
key, val, err = cursor.SeekExact([]byte("BBBB"))
require.Nil(t, err)
assert.Nil(t, key)
assert.Nil(t, val)
}
func TestFirstAfterClearBucket(t *testing.T) {
rwTx, err := New().BeginRw(context.Background())
require.NoError(t, err)
initializeDbNonDupSort(rwTx)
batch := NewMemoryBatch(rwTx, "")
defer batch.Close()
err = batch.ClearBucket(kv.HashedAccounts)
require.Nil(t, err)
err = batch.Put(kv.HashedAccounts, []byte("BBBB"), []byte("value5"))
require.Nil(t, err)
cursor, err := batch.Cursor(kv.HashedAccounts)
require.NoError(t, err)
key, val, err := cursor.First()
require.Nil(t, err)
assert.Equal(t, []byte("BBBB"), key)
assert.Equal(t, []byte("value5"), val)
key, val, err = cursor.Next()
require.Nil(t, err)
assert.Nil(t, key)
assert.Nil(t, val)
}
func TestIncReadSequence(t *testing.T) {
rwTx, err := New().BeginRw(context.Background())
require.NoError(t, err)
initializeDbNonDupSort(rwTx)
batch := NewMemoryBatch(rwTx, "")
defer batch.Close()
_, err = batch.IncrementSequence(kv.HashedAccounts, uint64(12))
require.Nil(t, err)
val, err := batch.ReadSequence(kv.HashedAccounts)
require.Nil(t, err)
require.Equal(t, val, uint64(12))
}
func initializeDbDupSort(rwTx kv.RwTx) {
rwTx.Put(kv.AccountChangeSet, []byte("key1"), []byte("value1.1"))
rwTx.Put(kv.AccountChangeSet, []byte("key3"), []byte("value3.1"))
rwTx.Put(kv.AccountChangeSet, []byte("key1"), []byte("value1.3"))
rwTx.Put(kv.AccountChangeSet, []byte("key3"), []byte("value3.3"))
}
func TestNext(t *testing.T) {
rwTx, err := New().BeginRw(context.Background())
require.NoError(t, err)
initializeDbDupSort(rwTx)
batch := NewMemoryBatch(rwTx, "")
defer batch.Close()
batch.Put(kv.AccountChangeSet, []byte("key1"), []byte("value1.2"))
cursor, err := batch.CursorDupSort(kv.AccountChangeSet)
require.NoError(t, err)
k, v, err := cursor.First()
require.Nil(t, err)
assert.Equal(t, []byte("key1"), k)
assert.Equal(t, []byte("value1.1"), v)
k, v, err = cursor.Next()
require.Nil(t, err)
assert.Equal(t, []byte("key1"), k)
assert.Equal(t, []byte("value1.2"), v)
k, v, err = cursor.Next()
require.Nil(t, err)
assert.Equal(t, []byte("key1"), k)
assert.Equal(t, []byte("value1.3"), v)
k, v, err = cursor.Next()
require.Nil(t, err)
assert.Equal(t, []byte("key3"), k)
assert.Equal(t, []byte("value3.1"), v)
k, v, err = cursor.Next()
require.Nil(t, err)
assert.Equal(t, []byte("key3"), k)
assert.Equal(t, []byte("value3.3"), v)
k, v, err = cursor.Next()
require.Nil(t, err)
assert.Nil(t, k)
assert.Nil(t, v)
}
func TestNextNoDup(t *testing.T) {
rwTx, err := New().BeginRw(context.Background())
require.NoError(t, err)
initializeDbDupSort(rwTx)
batch := NewMemoryBatch(rwTx, "")
defer batch.Close()
batch.Put(kv.AccountChangeSet, []byte("key2"), []byte("value2.1"))
batch.Put(kv.AccountChangeSet, []byte("key2"), []byte("value2.2"))
cursor, err := batch.CursorDupSort(kv.AccountChangeSet)
require.NoError(t, err)
k, _, err := cursor.First()
require.Nil(t, err)
assert.Equal(t, []byte("key1"), k)
k, _, err = cursor.NextNoDup()
require.Nil(t, err)
assert.Equal(t, []byte("key2"), k)
k, _, err = cursor.NextNoDup()
require.Nil(t, err)
assert.Equal(t, []byte("key3"), k)
}
func TestDeleteCurrentDuplicates(t *testing.T) {
rwTx, err := New().BeginRw(context.Background())
require.NoError(t, err)
initializeDbDupSort(rwTx)
batch := NewMemoryBatch(rwTx, "")
defer batch.Close()
cursor, err := batch.RwCursorDupSort(kv.AccountChangeSet)
require.NoError(t, err)
require.NoError(t, cursor.Put([]byte("key3"), []byte("value3.2")))
key, _, err := cursor.SeekExact([]byte("key3"))
require.NoError(t, err)
require.Equal(t, []byte("key3"), key)
require.NoError(t, cursor.DeleteCurrentDuplicates())
require.NoError(t, batch.Flush(rwTx))
var keys []string
var values []string
err = rwTx.ForEach(kv.AccountChangeSet, nil, func(k, v []byte) error {
keys = append(keys, string(k))
values = append(values, string(v))
return nil
})
require.NoError(t, err)
require.Equal(t, []string{"key1", "key1"}, keys)
require.Equal(t, []string{"value1.1", "value1.3"}, values)
}
func TestSeekBothRange(t *testing.T) {
rwTx, err := New().BeginRw(context.Background())
require.NoError(t, err)
rwTx.Put(kv.AccountChangeSet, []byte("key1"), []byte("value1.1"))
rwTx.Put(kv.AccountChangeSet, []byte("key3"), []byte("value3.3"))
batch := NewMemoryBatch(rwTx, "")
defer batch.Close()
cursor, err := batch.RwCursorDupSort(kv.AccountChangeSet)
require.NoError(t, err)
require.NoError(t, cursor.Put([]byte("key3"), []byte("value3.1")))
require.NoError(t, cursor.Put([]byte("key1"), []byte("value1.3")))
v, err := cursor.SeekBothRange([]byte("key2"), []byte("value1.2"))
require.NoError(t, err)
// SeekBothRange does exact match of the key, but range match of the value, so we get nil here
require.Nil(t, v)
v, err = cursor.SeekBothRange([]byte("key3"), []byte("value3.2"))
require.NoError(t, err)
require.Equal(t, "value3.3", string(v))
}
func initializeDbAutoConversion(rwTx kv.RwTx) {
rwTx.Put(kv.PlainState, []byte("A"), []byte("0"))
rwTx.Put(kv.PlainState, []byte("A..........................._______________________________A"), []byte("1"))
rwTx.Put(kv.PlainState, []byte("A..........................._______________________________C"), []byte("2"))
rwTx.Put(kv.PlainState, []byte("B"), []byte("8"))
rwTx.Put(kv.PlainState, []byte("C"), []byte("9"))
rwTx.Put(kv.PlainState, []byte("D..........................._______________________________A"), []byte("3"))
rwTx.Put(kv.PlainState, []byte("D..........................._______________________________C"), []byte("4"))
}
func TestAutoConversion(t *testing.T) {
rwTx, err := New().BeginRw(context.Background())
require.NoError(t, err)
initializeDbAutoConversion(rwTx)
batch := NewMemoryBatch(rwTx, "")
defer batch.Close()
c, err := batch.RwCursor(kv.PlainState)
require.NoError(t, err)
// key length conflict
require.Error(t, c.Put([]byte("A..........................."), []byte("?")))
require.NoError(t, c.Delete([]byte("A..........................._______________________________A")))
require.NoError(t, c.Put([]byte("B"), []byte("7")))
require.NoError(t, c.Delete([]byte("C")))
require.NoError(t, c.Put([]byte("D..........................._______________________________C"), []byte("6")))
require.NoError(t, c.Put([]byte("D..........................._______________________________E"), []byte("5")))
k, v, err := c.First()
require.NoError(t, err)
assert.Equal(t, []byte("A"), k)
assert.Equal(t, []byte("0"), v)
k, v, err = c.Next()
require.NoError(t, err)
assert.Equal(t, []byte("A..........................._______________________________C"), k)
assert.Equal(t, []byte("2"), v)
k, v, err = c.Next()
require.NoError(t, err)
assert.Equal(t, []byte("B"), k)
assert.Equal(t, []byte("7"), v)
k, v, err = c.Next()
require.NoError(t, err)
assert.Equal(t, []byte("D..........................._______________________________A"), k)
assert.Equal(t, []byte("3"), v)
k, v, err = c.Next()
require.NoError(t, err)
assert.Equal(t, []byte("D..........................._______________________________C"), k)
assert.Equal(t, []byte("6"), v)
k, v, err = c.Next()
require.NoError(t, err)
assert.Equal(t, []byte("D..........................._______________________________E"), k)
assert.Equal(t, []byte("5"), v)
k, v, err = c.Next()
require.NoError(t, err)
assert.Nil(t, k)
assert.Nil(t, v)
}
func TestAutoConversionDelete(t *testing.T) {
rwTx, err := New().BeginRw(context.Background())
require.NoError(t, err)
initializeDbAutoConversion(rwTx)
batch := NewMemoryBatch(rwTx, "")
defer batch.Close()
c, err := batch.RwCursor(kv.PlainState)
require.NoError(t, err)
require.NoError(t, c.Delete([]byte("A..........................._______________________________A")))
require.NoError(t, c.Delete([]byte("A..........................._______________________________C")))
require.NoError(t, c.Delete([]byte("B")))
require.NoError(t, c.Delete([]byte("C")))
k, v, err := c.First()
require.NoError(t, err)
assert.Equal(t, []byte("A"), k)
assert.Equal(t, []byte("0"), v)
k, v, err = c.Next()
require.NoError(t, err)
assert.Equal(t, []byte("D..........................._______________________________A"), k)
assert.Equal(t, []byte("3"), v)
k, v, err = c.Next()
require.NoError(t, err)
assert.Equal(t, []byte("D..........................._______________________________C"), k)
assert.Equal(t, []byte("4"), v)
k, v, err = c.Next()
require.NoError(t, err)
assert.Nil(t, k)
assert.Nil(t, v)
}
func TestAutoConversionSeekBothRange(t *testing.T) {
rwTx, err := New().BeginRw(context.Background())
require.NoError(t, err)
initializeDbAutoConversion(rwTx)
batch := NewMemoryBatch(rwTx, "")
defer batch.Close()
c, err := batch.RwCursorDupSort(kv.PlainState)
require.NoError(t, err)
require.NoError(t, c.Delete([]byte("A..........................._______________________________A")))
require.NoError(t, c.Put([]byte("D..........................._______________________________C"), []byte("6")))
require.NoError(t, c.Put([]byte("D..........................._______________________________E"), []byte("5")))
v, err := c.SeekBothRange([]byte("A..........................."), []byte("_______________________________A"))
require.NoError(t, err)
assert.Equal(t, []byte("_______________________________C2"), v)
_, v, err = c.NextDup()
require.NoError(t, err)
assert.Nil(t, v)
v, err = c.SeekBothRange([]byte("A..........................."), []byte("_______________________________X"))
require.NoError(t, err)
assert.Nil(t, v)
v, err = c.SeekBothRange([]byte("B..........................."), []byte(""))
require.NoError(t, err)
assert.Nil(t, v)
v, err = c.SeekBothRange([]byte("C..........................."), []byte(""))
require.NoError(t, err)
assert.Nil(t, v)
v, err = c.SeekBothRange([]byte("D..........................."), []byte(""))
require.NoError(t, err)
assert.Equal(t, []byte("_______________________________A3"), v)
_, v, err = c.NextDup()
require.NoError(t, err)
assert.Equal(t, []byte("_______________________________C6"), v)
_, v, err = c.NextDup()
require.NoError(t, err)
assert.Equal(t, []byte("_______________________________E5"), v)
_, v, err = c.NextDup()
require.NoError(t, err)
assert.Nil(t, v)
v, err = c.SeekBothRange([]byte("X..........................."), []byte("_______________________________Y"))
require.NoError(t, err)
assert.Nil(t, v)
}