erigon-pulse/core/generate_index_test.go
Alex Sharov b3f1915d09
ChangeSets dupsort (#1342)
* change_set_dup

* change_set_dup

* change_set_dup

* change_set_dup

* change_set_dup

* change_set_dup

* change_set_dup

* change_set_dup

* change_set_dup

* change_set_dup

* change_set_dup

* change_set_dup

* change_set_dup

* change_set_dup

* change_set_dup

* change_set_dup

* change_set_dup

* change_set_dup

* change_set_dup

* change_set_dup

* change_set_dup

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* working version

* aa

* aa

* aa

* aa

* aa

* aa

* aa

* aa

* aa

* aa

* aa

* aa

* aa

* aa

* aa

* squash

* squash

* fix

* fix

* fix

* fix

* fix

* fix

* fix

* fix

* fix

* fix

* fix

* history_early_stop

* history_early_stop

* vmConfig with ReadOnly false

* auto_increment

* auto_increment

* rebase master

Co-authored-by: Alexey Akhunov <akhounov@gmail.com>
2020-11-16 12:08:28 +00:00

317 lines
10 KiB
Go

package core
import (
"encoding/binary"
"fmt"
"os"
"reflect"
"sort"
"strconv"
"testing"
"github.com/ledgerwatch/turbo-geth/common/changeset"
"github.com/ledgerwatch/turbo-geth/common"
"github.com/ledgerwatch/turbo-geth/common/dbutils"
"github.com/ledgerwatch/turbo-geth/crypto"
"github.com/ledgerwatch/turbo-geth/ethdb"
"github.com/ledgerwatch/turbo-geth/log"
)
func TestIndexGenerator_GenerateIndex_SimpleCase(t *testing.T) {
log.Root().SetHandler(log.LvlFilterHandler(log.LvlInfo, log.StreamHandler(os.Stderr, log.TerminalFormat(true))))
test := func(blocksNum int, csBucket string) func(t *testing.T) {
return func(t *testing.T) {
db := ethdb.NewMemDatabase()
defer db.Close()
ig := NewIndexGenerator("logPrefix", db, make(chan struct{}))
csInfo, ok := changeset.Mapper[csBucket]
if !ok {
t.Fatal("incorrect cs bucket")
}
addrs, expecedIndexes := generateTestData(t, db, csBucket, blocksNum)
ig.ChangeSetBufSize = 16 * 1024
err := ig.GenerateIndex(0, uint64(blocksNum), csBucket, "")
if err != nil {
t.Fatal(err)
}
checkIndex(t, db, csInfo.IndexBucket, addrs[0], 0, expecedIndexes[string(addrs[0])][0])
checkIndex(t, db, csInfo.IndexBucket, addrs[0], 999, expecedIndexes[string(addrs[0])][0])
checkIndex(t, db, csInfo.IndexBucket, addrs[0], 1000, expecedIndexes[string(addrs[0])][1])
checkIndex(t, db, csInfo.IndexBucket, addrs[0], 1999, expecedIndexes[string(addrs[0])][1])
checkIndex(t, db, csInfo.IndexBucket, addrs[0], 2000, expecedIndexes[string(addrs[0])][2])
checkIndex(t, db, csInfo.IndexBucket, addrs[1], 0, expecedIndexes[string(addrs[1])][0])
checkIndex(t, db, csInfo.IndexBucket, addrs[1], 2000, expecedIndexes[string(addrs[1])][1])
checkIndex(t, db, csInfo.IndexBucket, addrs[2], 0, expecedIndexes[string(addrs[2])][0])
//check last chunk
lastChunkCheck(t, db, csInfo.IndexBucket, addrs[0], expecedIndexes[string(addrs[0])][2])
lastChunkCheck(t, db, csInfo.IndexBucket, addrs[1], expecedIndexes[string(addrs[1])][1])
lastChunkCheck(t, db, csInfo.IndexBucket, addrs[2], expecedIndexes[string(addrs[2])][0])
}
}
t.Run("account plain state", test(2100, dbutils.PlainAccountChangeSetBucket))
t.Run("storage plain state", test(2100, dbutils.PlainStorageChangeSetBucket))
}
func TestIndexGenerator_Truncate(t *testing.T) {
log.Root().SetHandler(log.LvlFilterHandler(log.LvlInfo, log.StreamHandler(os.Stderr, log.TerminalFormat(true))))
buckets := []string{dbutils.PlainAccountChangeSetBucket, dbutils.PlainStorageChangeSetBucket}
for i := range buckets {
csbucket := buckets[i]
db := ethdb.NewMemDatabase()
hashes, expected := generateTestData(t, db, csbucket, 2100)
mp := changeset.Mapper[csbucket]
indexBucket := mp.IndexBucket
ig := NewIndexGenerator("logPrefix", db, make(chan struct{}))
err := ig.GenerateIndex(0, uint64(2100), csbucket, "")
if err != nil {
t.Fatal(err)
}
reduceSlice := func(arr []uint64, timestamtTo uint64) []uint64 {
pos := sort.Search(len(arr), func(i int) bool {
return arr[i] > timestamtTo
})
return arr[:pos]
}
t.Run("truncate to 2050 "+csbucket, func(t *testing.T) {
expected[string(hashes[0])][2] = reduceSlice(expected[string(hashes[0])][2], 2050)
expected[string(hashes[1])][1] = reduceSlice(expected[string(hashes[1])][1], 2050)
expected[string(hashes[2])][0] = reduceSlice(expected[string(hashes[2])][0], 2050)
err = ig.Truncate(2050, csbucket)
if err != nil {
t.Fatal(err)
}
checkIndex(t, db, indexBucket, hashes[0], 2030, expected[string(hashes[0])][2])
checkIndex(t, db, indexBucket, hashes[1], 2030, expected[string(hashes[1])][1])
checkIndex(t, db, indexBucket, hashes[2], 2030, expected[string(hashes[2])][0])
checkIndex(t, db, indexBucket, hashes[0], 1999, expected[string(hashes[0])][1])
checkIndex(t, db, indexBucket, hashes[1], 999, expected[string(hashes[1])][0])
})
t.Run("truncate to 2000 "+string(csbucket), func(t *testing.T) {
expected[string(hashes[0])][2] = reduceSlice(expected[string(hashes[0])][2], 2000)
expected[string(hashes[1])][1] = reduceSlice(expected[string(hashes[1])][1], 2000)
expected[string(hashes[2])][0] = reduceSlice(expected[string(hashes[2])][0], 2000)
err = ig.Truncate(2000, csbucket)
if err != nil {
t.Fatal(err)
}
checkIndex(t, db, indexBucket, hashes[0], 2000, expected[string(hashes[0])][2])
checkIndex(t, db, indexBucket, hashes[1], 2000, expected[string(hashes[1])][1])
checkIndex(t, db, indexBucket, hashes[2], expected[string(hashes[2])][0][len(expected[string(hashes[2])][0])-1], expected[string(hashes[2])][0])
})
t.Run("truncate to 1999 "+string(csbucket), func(t *testing.T) {
err = ig.Truncate(1999, csbucket)
if err != nil {
t.Fatal(err)
}
checkIndex(t, db, indexBucket, hashes[0], 1999, expected[string(hashes[0])][1])
checkIndex(t, db, indexBucket, hashes[1], 1998, expected[string(hashes[1])][0])
checkIndex(t, db, indexBucket, hashes[2], 1998, expected[string(hashes[2])][0])
_, err = db.GetIndexChunk(csbucket, hashes[0], 2000)
if err != ethdb.ErrKeyNotFound {
t.Fatal()
}
_, err = db.GetIndexChunk(csbucket, hashes[1], 2000)
if err != ethdb.ErrKeyNotFound {
t.Fatal()
}
})
t.Run("truncate to 999 "+string(csbucket), func(t *testing.T) {
expected[string(hashes[1])][0] = reduceSlice(expected[string(hashes[1])][0], 999)
expected[string(hashes[2])][0] = reduceSlice(expected[string(hashes[2])][0], 999)
err = ig.Truncate(999, csbucket)
if err != nil {
t.Fatal(err)
}
checkIndex(t, db, indexBucket, hashes[0], 999, expected[string(hashes[0])][0])
checkIndex(t, db, indexBucket, hashes[1], 998, expected[string(hashes[1])][0])
checkIndex(t, db, indexBucket, hashes[2], 999, expected[string(hashes[2])][0])
_, err = db.GetIndexChunk(csbucket, hashes[0], 1000)
if err != ethdb.ErrKeyNotFound {
t.Fatal()
}
_, err = db.GetIndexChunk(csbucket, hashes[1], 1000)
if err != ethdb.ErrKeyNotFound {
t.Fatal()
}
})
db.Close()
}
}
func generateTestData(t *testing.T, db ethdb.Database, csBucket string, numOfBlocks int) ([][]byte, map[string][][]uint64) { //nolint
csInfo, ok := changeset.Mapper[csBucket]
if !ok {
t.Fatal("incorrect cs bucket")
}
var isPlain bool
if dbutils.PlainStorageChangeSetBucket == csBucket || dbutils.PlainAccountChangeSetBucket == csBucket {
isPlain = true
}
addrs, err := generateAddrs(3, isPlain)
if err != nil {
t.Fatal(err)
}
if dbutils.StorageChangeSetBucket == csBucket || dbutils.PlainStorageChangeSetBucket == csBucket {
keys, innerErr := generateAddrs(3, false)
if innerErr != nil {
t.Fatal(innerErr)
}
defaultIncarnation := make([]byte, 8)
binary.BigEndian.PutUint64(defaultIncarnation, uint64(1))
for i := range addrs {
addrs[i] = append(addrs[i], defaultIncarnation...)
addrs[i] = append(addrs[i], keys[i]...)
}
}
expected1 := make([]dbutils.HistoryIndexBytes, 0)
expected1 = append(expected1, dbutils.NewHistoryIndex())
expected2 := make([]dbutils.HistoryIndexBytes, 0)
expected2 = append(expected2, dbutils.NewHistoryIndex())
expected3 := make([]dbutils.HistoryIndexBytes, 0)
expected3 = append(expected3, dbutils.NewHistoryIndex())
for i := 0; i < numOfBlocks; i++ {
cs := csInfo.New()
err = cs.Add(addrs[0], []byte(strconv.Itoa(i)))
if err != nil {
t.Fatal(err)
}
if dbutils.CheckNewIndexChunk(expected1[len(expected1)-1], uint64(i)) {
expected1 = append(expected1, dbutils.NewHistoryIndex())
}
expected1[len(expected1)-1] = expected1[len(expected1)-1].Append(uint64(i), false)
if i%2 == 0 {
err = cs.Add(addrs[1], []byte(strconv.Itoa(i)))
if err != nil {
t.Fatal(err)
}
if dbutils.CheckNewIndexChunk(expected2[len(expected2)-1], uint64(i)) {
expected2 = append(expected2, dbutils.NewHistoryIndex())
}
expected2[len(expected2)-1] = expected2[len(expected2)-1].Append(uint64(i), false)
}
if i%3 == 0 {
err = cs.Add(addrs[2], []byte(strconv.Itoa(i)))
if err != nil {
t.Fatal(err)
}
if dbutils.CheckNewIndexChunk(expected3[len(expected3)-1], uint64(i)) {
expected3 = append(expected3, dbutils.NewHistoryIndex())
}
expected3[len(expected3)-1] = expected3[len(expected3)-1].Append(uint64(i), false)
}
err = csInfo.Encode(uint64(i), cs, func(k, v []byte) error {
return db.Append(csBucket, k, v)
})
if err != nil {
t.Fatal(err)
}
}
res := make([][]uint64, len(expected1))
for i := range expected1 {
res[i], _, err = expected1[i].Decode()
if err != nil {
t.Fatal(err)
}
}
res2 := make([][]uint64, len(expected2))
for i := range expected2 {
res2[i], _, err = expected2[i].Decode()
if err != nil {
t.Fatal(err)
}
}
res3 := make([][]uint64, len(expected3))
for i := range expected3 {
res3[i], _, err = expected3[i].Decode()
if err != nil {
t.Fatal(err)
}
}
return addrs, map[string][][]uint64{
string(addrs[0]): res,
string(addrs[1]): res2,
string(addrs[2]): res3,
}
}
func checkIndex(t *testing.T, db ethdb.Database, bucket string, addrHash []byte, chunkBlock uint64, expected []uint64) {
t.Helper()
b, err := db.GetIndexChunk(bucket, addrHash, chunkBlock)
if err != nil {
t.Fatal(err, common.Bytes2Hex(addrHash), chunkBlock)
}
val, _, err := dbutils.HistoryIndexBytes(b).Decode()
if err != nil {
t.Fatal(err)
}
if !reflect.DeepEqual(val, expected) {
fmt.Println("get", val)
fmt.Println("expected", expected)
t.Fatal()
}
}
func lastChunkCheck(t *testing.T, db ethdb.Database, bucket string, key []byte, expected []uint64) {
t.Helper()
v, err := db.Get(bucket, dbutils.CurrentChunkKey(key))
if err != nil {
t.Fatal(err, dbutils.CurrentChunkKey(key))
}
val, _, err := dbutils.HistoryIndexBytes(v).Decode()
if err != nil {
t.Fatal(err)
}
if !reflect.DeepEqual(val, expected) {
fmt.Println(val)
fmt.Println(expected)
t.Fatal()
}
}
func generateAddrs(numOfAddrs int, isPlain bool) ([][]byte, error) {
addrs := make([][]byte, numOfAddrs)
for i := 0; i < numOfAddrs; i++ {
key1, err := crypto.GenerateKey()
if err != nil {
return nil, err
}
addr := crypto.PubkeyToAddress(key1.PublicKey)
if isPlain {
addrs[i] = addr.Bytes()
continue
}
hash, err := common.HashData(addr.Bytes())
if err != nil {
return nil, err
}
addrs[i] = hash.Bytes()
}
return addrs, nil
}