erigon-pulse/ethdb/mutation_test.go
b00ris 2b74b43678 History index encoding
save state

save state

remove old history index

fix lint

compress index
2020-02-24 23:29:46 +03:00

567 lines
15 KiB
Go

package ethdb
import (
"bytes"
"github.com/ledgerwatch/turbo-geth/common/changeset"
"math/big"
"math/rand"
"reflect"
"sort"
"testing"
"github.com/davecgh/go-spew/spew"
"github.com/ledgerwatch/turbo-geth/common"
"github.com/ledgerwatch/turbo-geth/common/dbutils"
"github.com/ledgerwatch/turbo-geth/common/debug"
"github.com/ledgerwatch/turbo-geth/core/types/accounts"
"github.com/ledgerwatch/turbo-geth/crypto"
"github.com/stretchr/testify/assert"
)
func TestMutation_DeleteTimestamp(t *testing.T) {
db := NewMemDatabase()
mutDB := db.NewBatch()
acc := make([]*accounts.Account, 10)
addrHashes := make([]common.Hash, 10)
for i := range acc {
acc[i], addrHashes[i] = randomAccount(t)
b := make([]byte, acc[i].EncodingLengthForStorage())
acc[i].EncodeForStorage(b)
err := mutDB.PutS(dbutils.AccountsHistoryBucket, addrHashes[i].Bytes(), b, 1, false)
if err != nil {
t.Fatal(err)
}
}
_, err := mutDB.Commit()
if err != nil {
t.Fatal(err)
}
csData, err := db.Get(dbutils.AccountChangeSetBucket, dbutils.EncodeTimestamp(1))
if err != nil {
t.Fatal(err)
}
if changeset.Len(csData) != 10 {
t.FailNow()
}
if debug.IsThinHistory() {
csData, err = db.Get(dbutils.AccountsHistoryBucket, addrHashes[0].Bytes())
if err != nil {
t.Fatal(err)
}
index := dbutils.WrapHistoryIndex(csData)
parsed, innerErr := index.Decode()
if innerErr != nil {
t.Fatal(innerErr)
}
if parsed[0] != 1 {
t.Fatal("incorrect block num")
}
} else {
compositeKey, _ := dbutils.CompositeKeySuffix(addrHashes[0].Bytes(), 1)
_, innerErr := db.Get(dbutils.AccountsHistoryBucket, compositeKey)
if innerErr != nil {
t.Fatal(innerErr)
}
}
err = mutDB.DeleteTimestamp(1)
if err != nil {
t.Fatal(err)
}
_, err = mutDB.Commit()
if err != nil {
t.Fatal(err)
}
_, err = db.Get(dbutils.AccountChangeSetBucket, dbutils.EncodeTimestamp(1))
if err != ErrKeyNotFound {
t.Fatal("changeset must be deleted")
}
if debug.IsThinHistory() {
_, err = db.Get(dbutils.AccountsHistoryBucket, addrHashes[0].Bytes())
if err != ErrKeyNotFound {
t.Fatal("account must be deleted")
}
} else {
compositeKey, _ := dbutils.CompositeKeySuffix(addrHashes[0].Bytes(), 1)
_, err = db.Get(dbutils.AccountsHistoryBucket, compositeKey)
if err != ErrKeyNotFound {
t.Fatal("account must be deleted")
}
}
}
func TestMutationCommit(t *testing.T) {
if debug.IsThinHistory() {
t.Skip()
}
db := NewMemDatabase()
mutDB := db.NewBatch()
numOfAccounts := 5
numOfStateKeys := 5
addrHashes, accState, accStateStorage, accHistory, accHistoryStateStorage := generateAccountsWithStorageAndHistory(t, mutDB, numOfAccounts, numOfStateKeys)
_, commitErr := mutDB.Commit()
if commitErr != nil {
t.Fatal(commitErr)
}
for i, addrHash := range addrHashes {
b, err := db.Get(dbutils.AccountsBucket, addrHash.Bytes())
if err != nil {
t.Fatal("error on get account", i, err)
}
acc := accounts.NewAccount()
err = acc.DecodeForStorage(b)
if err != nil {
t.Fatal("error on get account", i, err)
}
if !accState[i].Equals(&acc) {
spew.Dump("got", acc)
spew.Dump("expected", accState[i])
t.Fatal("Accounts not equals")
}
compositeKey, _ := dbutils.CompositeKeySuffix(addrHash.Bytes(), 1)
b, err = db.Get(dbutils.AccountsHistoryBucket, compositeKey)
if err != nil {
t.Fatal("error on get account", i, err)
}
acc = accounts.NewAccount()
err = acc.DecodeForStorage(b)
if err != nil {
t.Fatal("error on get account", i, err)
}
if !accHistory[i].Equals(&acc) {
spew.Dump("got", acc)
spew.Dump("expected", accState[i])
t.Fatal("Accounts not equals")
}
resAccStorage := make(map[common.Hash]common.Hash)
err = db.Walk(dbutils.StorageBucket, dbutils.GenerateStoragePrefix(addrHash, acc.Incarnation), common.HashLength+8, func(k, v []byte) (b bool, e error) {
resAccStorage[common.BytesToHash(k[common.HashLength+8:])] = common.BytesToHash(v)
return true, nil
})
if err != nil {
t.Fatal("error on get account storage", i, err)
}
if !reflect.DeepEqual(resAccStorage, accStateStorage[i]) {
spew.Dump("res", resAccStorage)
spew.Dump("expected", accHistoryStateStorage[i])
t.Log("incorrect storage", i)
}
resAccStorage = make(map[common.Hash]common.Hash)
err = db.Walk(dbutils.StorageHistoryBucket, dbutils.GenerateStoragePrefix(addrHash, acc.Incarnation), common.HashLength+8, func(k, v []byte) (b bool, e error) {
resAccStorage[common.BytesToHash(k[common.HashLength+8:common.HashLength+8+common.HashLength])] = common.BytesToHash(v)
return true, nil
})
if err != nil {
t.Fatal("error on get account storage", i, err)
}
if !reflect.DeepEqual(resAccStorage, accHistoryStateStorage[i]) {
spew.Dump("res", resAccStorage)
spew.Dump("expected", accHistoryStateStorage[i])
t.Fatal("incorrect history storage", i)
}
}
csData, err := db.Get(dbutils.AccountChangeSetBucket, dbutils.EncodeTimestamp(1))
if err != nil {
t.Fatal(err)
}
expectedChangeSet := changeset.NewAccountChangeSet()
for i := range addrHashes {
b := make([]byte, accHistory[i].EncodingLengthForStorage())
accHistory[i].EncodeForStorage(b)
err = expectedChangeSet.Add(addrHashes[i].Bytes(), b)
if err != nil {
t.Fatal(err)
}
}
sort.Sort(expectedChangeSet)
expectedData, err := changeset.EncodeChangeSet(expectedChangeSet)
assert.NoError(t, err)
if !bytes.Equal(csData, expectedData) {
spew.Dump("res", csData)
spew.Dump("expected", expectedData)
t.Fatal("incorrect account changeset")
}
csData, err = db.Get(dbutils.StorageChangeSetBucket, dbutils.EncodeTimestamp(1))
if err != nil {
t.Fatal(err)
}
if changeset.Len(csData) != numOfAccounts*numOfStateKeys {
t.FailNow()
}
expectedChangeSet = changeset.NewStorageChangeSet()
for i, addrHash := range addrHashes {
for j := 0; j < numOfStateKeys; j++ {
key := common.Hash{uint8(i*100 + j)}
value := common.Hash{uint8(10 + j)}
if err := expectedChangeSet.Add(dbutils.GenerateCompositeStorageKey(addrHash, accHistory[i].Incarnation, key), value.Bytes()); err != nil {
t.Fatal(err)
}
}
}
sort.Sort(expectedChangeSet)
expectedData, err = changeset.EncodeChangeSet(expectedChangeSet)
if debug.IsThinHistory() {
expectedData, err = changeset.EncodeStorage(expectedChangeSet)
}
assert.NoError(t, err)
if !bytes.Equal(csData, expectedData) {
spew.Dump("res", csData)
spew.Dump("expected", expectedData)
t.Fatal("incorrect storage changeset")
}
}
func TestMutationCommitThinHistory(t *testing.T) {
if !debug.IsThinHistory() {
t.Skip()
}
db := NewMemDatabase()
mutDB := db.NewBatch()
numOfAccounts := 5
numOfStateKeys := 5
addrHashes, accState, accStateStorage, accHistory, accHistoryStateStorage := generateAccountsWithStorageAndHistory(t, mutDB, numOfAccounts, numOfStateKeys)
_, commitErr := mutDB.Commit()
if commitErr != nil {
t.Fatal(commitErr)
}
for i, addrHash := range addrHashes {
b, err := db.Get(dbutils.AccountsBucket, addrHash.Bytes())
if err != nil {
t.Fatal("error on get account", i, err)
}
acc := accounts.NewAccount()
err = acc.DecodeForStorage(b)
if err != nil {
t.Fatal("error on get account", i, err)
}
if !accState[i].Equals(&acc) {
spew.Dump("got", acc)
spew.Dump("expected", accState[i])
t.Fatal("Accounts not equals")
}
b, err = db.Get(dbutils.AccountsHistoryBucket, addrHash.Bytes())
if err != nil {
t.Fatal("error on get account", i, err)
}
index := dbutils.WrapHistoryIndex(b)
parsedIndex, err := index.Decode()
if err != nil {
t.Fatal("error on get account", i, err)
}
if parsedIndex[0] != 1 && index.Len() != 1 {
t.Fatal("incorrect history index")
}
resAccStorage := make(map[common.Hash]common.Hash)
err = db.Walk(dbutils.StorageBucket, dbutils.GenerateStoragePrefix(addrHash, acc.Incarnation), common.HashLength+8, func(k, v []byte) (b bool, e error) {
resAccStorage[common.BytesToHash(k[common.HashLength+8:])] = common.BytesToHash(v)
return true, nil
})
if err != nil {
t.Fatal("error on get account storage", i, err)
}
if !reflect.DeepEqual(resAccStorage, accStateStorage[i]) {
spew.Dump("res", resAccStorage)
spew.Dump("expected", accStateStorage[i])
t.Fatal("incorrect storage", i)
}
for k, v := range accHistoryStateStorage[i] {
res, err := db.GetAsOf(dbutils.StorageBucket, dbutils.StorageHistoryBucket, dbutils.GenerateCompositeStorageKey(addrHash, acc.Incarnation, k), 1)
if err != nil {
t.Fatal(err)
}
resultHash := common.BytesToHash(res)
if resultHash != v {
t.Fatal("incorrect storage history for ", addrHash.String(), v, resultHash)
}
}
}
csData, err := db.Get(dbutils.AccountChangeSetBucket, dbutils.EncodeTimestamp(1))
if err != nil {
t.Fatal(err)
}
expectedChangeSet := changeset.NewAccountChangeSet()
for i := range addrHashes {
b := make([]byte, accHistory[i].EncodingLengthForStorage())
accHistory[i].EncodeForStorage(b)
innerErr := expectedChangeSet.Add(addrHashes[i].Bytes(), b)
if innerErr != nil {
t.Fatal(innerErr)
}
}
expectedData, err := changeset.EncodeChangeSet(expectedChangeSet)
assert.NoError(t, err)
if !bytes.Equal(csData, expectedData) {
t.Fatal("incorrect changeset")
}
csData, err = db.Get(dbutils.StorageChangeSetBucket, dbutils.EncodeTimestamp(1))
if err != nil {
t.Fatal(err)
}
if changeset.Len(csData) != numOfAccounts*numOfStateKeys {
t.FailNow()
}
expectedChangeSet = changeset.NewStorageChangeSet()
for i, addrHash := range addrHashes {
for j := 0; j < numOfStateKeys; j++ {
key := common.Hash{uint8(i*100 + j)}
value := common.Hash{uint8(10 + j)}
err := expectedChangeSet.Add(dbutils.GenerateCompositeStorageKey(addrHash, accHistory[i].Incarnation, key), value.Bytes())
if err != nil {
t.Fatal(err)
}
}
}
expectedData, err = changeset.EncodeStorage(expectedChangeSet)
assert.NoError(t, err)
if !bytes.Equal(csData, expectedData) {
spew.Dump("res", csData)
spew.Dump("expected", expectedData)
t.Fatal("incorrect changeset")
}
}
func generateAccountsWithStorageAndHistory(t *testing.T, db Putter, numOfAccounts, numOfStateKeys int) ([]common.Hash, []*accounts.Account, []map[common.Hash]common.Hash, []*accounts.Account, []map[common.Hash]common.Hash) {
t.Helper()
accHistory := make([]*accounts.Account, numOfAccounts)
accState := make([]*accounts.Account, numOfAccounts)
accStateStorage := make([]map[common.Hash]common.Hash, numOfAccounts)
accHistoryStateStorage := make([]map[common.Hash]common.Hash, numOfAccounts)
addrHashes := make([]common.Hash, numOfAccounts)
for i := range accHistory {
var b []byte
accHistory[i], addrHashes[i] = randomAccount(t)
accHistory[i].Balance = *big.NewInt(100)
accHistory[i].CodeHash = common.Hash{uint8(10 + i)}
accHistory[i].Root = common.Hash{uint8(10 + i)}
accHistory[i].Incarnation = uint64(i)
b = make([]byte, accHistory[i].EncodingLengthForStorage())
accHistory[i].EncodeForStorage(b)
err := db.PutS(dbutils.AccountsHistoryBucket, addrHashes[i].Bytes(), b, 1, false)
if err != nil {
t.Fatal(err)
}
accState[i] = accHistory[i].SelfCopy()
accState[i].Nonce++
accState[i].Balance = *big.NewInt(200)
b = make([]byte, accState[i].EncodingLengthForStorage())
accState[i].EncodeForStorage(b)
err = db.Put(dbutils.AccountsBucket, addrHashes[i].Bytes(), b)
if err != nil {
t.Fatal(err)
}
accStateStorage[i] = make(map[common.Hash]common.Hash)
accHistoryStateStorage[i] = make(map[common.Hash]common.Hash)
for j := 0; j < numOfStateKeys; j++ {
key := common.Hash{uint8(i*100 + j)}
value := common.Hash{uint8(j)}
accStateStorage[i][key] = value
err = db.Put(dbutils.StorageBucket, dbutils.GenerateCompositeStorageKey(addrHashes[i], accHistory[i].Incarnation, key), value.Bytes())
if err != nil {
t.Fatal(err)
}
newValue := common.Hash{uint8(10 + j)}
accHistoryStateStorage[i][key] = newValue
err = db.PutS(
dbutils.StorageHistoryBucket,
dbutils.GenerateCompositeStorageKey(addrHashes[i], accHistory[i].Incarnation, key),
newValue.Bytes(),
1,
false,
)
if err != nil {
t.Fatal(err)
}
}
}
return addrHashes, accState, accStateStorage, accHistory, accHistoryStateStorage
}
func TestMutation_GetAsOf(t *testing.T) {
db := NewMemDatabase()
mutDB := db.NewBatch()
acc, addrHash := randomAccount(t)
acc2 := acc.SelfCopy()
acc2.Nonce = 1
acc4 := acc.SelfCopy()
acc4.Nonce = 3
b := make([]byte, acc.EncodingLengthForStorage())
acc.EncodeForStorage(b)
err := db.Put(dbutils.AccountsBucket, addrHash.Bytes(), b)
if err != nil {
t.Fatal(err)
}
b = make([]byte, acc2.EncodingLengthForStorage())
acc2.EncodeForStorage(b)
err = db.PutS(dbutils.AccountsHistoryBucket, addrHash.Bytes(), b, 2, false)
if err != nil {
t.Fatal(err)
}
b = make([]byte, acc4.EncodingLengthForStorage())
acc4.EncodeForStorage(b)
err = db.PutS(dbutils.AccountsHistoryBucket, addrHash.Bytes(), b, 4, false)
if err != nil {
t.Fatal(err)
}
_, err = mutDB.Commit()
if err != nil {
t.Fatal(err)
}
b, err = db.Get(dbutils.AccountsBucket, addrHash.Bytes())
if err != nil {
t.Fatal(err)
}
resAcc := new(accounts.Account)
err = resAcc.DecodeForStorage(b)
if err != nil {
t.Fatal(err)
}
if !acc.Equals(resAcc) {
t.Fatal("Account from Get is incorrect")
}
b, err = db.GetAsOf(dbutils.AccountsBucket, dbutils.AccountsHistoryBucket, addrHash.Bytes(), 1)
if err != nil {
t.Fatal("incorrect value on block 1", err)
}
resAcc = new(accounts.Account)
err = resAcc.DecodeForStorage(b)
if err != nil {
t.Fatal(err)
}
if !acc2.Equals(resAcc) {
spew.Dump(resAcc)
t.Fatal("Account from GetAsOf(1) is incorrect")
}
b, err = db.GetAsOf(dbutils.AccountsBucket, dbutils.AccountsHistoryBucket, addrHash.Bytes(), 2)
if err != nil {
t.Fatal(err)
}
resAcc = new(accounts.Account)
err = resAcc.DecodeForStorage(b)
if err != nil {
t.Fatal(err)
}
if !acc2.Equals(resAcc) {
spew.Dump(resAcc)
t.Fatal("Account from GetAsOf(2) is incorrect")
}
b, err = db.GetAsOf(dbutils.AccountsBucket, dbutils.AccountsHistoryBucket, addrHash.Bytes(), 3)
if err != nil {
t.Fatal(err)
}
resAcc = new(accounts.Account)
err = resAcc.DecodeForStorage(b)
if err != nil {
t.Fatal(err)
}
if !acc4.Equals(resAcc) {
spew.Dump(resAcc)
t.Fatal("Account from GetAsOf(2) is incorrect")
}
b, err = db.GetAsOf(dbutils.AccountsBucket, dbutils.AccountsHistoryBucket, addrHash.Bytes(), 5)
if err != nil {
t.Fatal(err)
}
resAcc = new(accounts.Account)
err = resAcc.DecodeForStorage(b)
if err != nil {
t.Fatal(err)
}
if !acc.Equals(resAcc) {
t.Fatal("Account from GetAsOf(4) is incorrect")
}
b, err = db.GetAsOf(dbutils.AccountsBucket, dbutils.AccountsHistoryBucket, addrHash.Bytes(), 7)
if err != nil {
t.Fatal(err)
}
resAcc = new(accounts.Account)
err = resAcc.DecodeForStorage(b)
if err != nil {
t.Fatal(err)
}
if !acc.Equals(resAcc) {
t.Fatal("Account from GetAsOf(7) is incorrect")
}
}
func randomAccount(t *testing.T) (*accounts.Account, common.Hash) {
t.Helper()
key, err := crypto.GenerateKey()
if err != nil {
t.Fatal(err)
}
acc := accounts.NewAccount()
acc.Initialised = true
acc.Balance = *big.NewInt(rand.Int63())
addr := crypto.PubkeyToAddress(key.PublicKey)
addrHash, err := common.HashData(addr.Bytes())
if err != nil {
t.Fatal(err)
}
return &acc, addrHash
}