mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-19 00:54:12 +00:00
2b74b43678
save state save state remove old history index fix lint compress index
567 lines
15 KiB
Go
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
|
|
}
|