mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-04 01:54:28 +00:00
fbe7a2cb06
* Cutting incarnation out of history and changeset keys * Rollback the remove of incarnation only in changesets * Fix linter * Remove debug printing * Reinstate encoding of incarnations in the ChangeSets * Fix linter * Fix linter * Fix DefaultIncarnation * Fixes * Fix lint * Fix to history writing * Fix linter
780 lines
21 KiB
Go
780 lines
21 KiB
Go
package state
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"math/big"
|
|
"math/rand"
|
|
"reflect"
|
|
"sort"
|
|
"strconv"
|
|
"testing"
|
|
|
|
"github.com/davecgh/go-spew/spew"
|
|
"github.com/ledgerwatch/turbo-geth/common"
|
|
"github.com/ledgerwatch/turbo-geth/common/changeset"
|
|
"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/ledgerwatch/turbo-geth/ethdb"
|
|
"github.com/ledgerwatch/turbo-geth/trie"
|
|
"github.com/stretchr/testify/assert"
|
|
)
|
|
|
|
func TestMutation_DeleteTimestamp(t *testing.T) {
|
|
db := ethdb.NewMemDatabase()
|
|
mutDB := db.NewBatch()
|
|
|
|
acc := make([]*accounts.Account, 10)
|
|
addr := make([]common.Address, 10)
|
|
addrHashes := make([]common.Hash, 10)
|
|
tds := NewTrieDbState(common.Hash{}, mutDB, 1)
|
|
blockWriter := tds.DbStateWriter()
|
|
ctx := context.Background()
|
|
emptyAccount := accounts.NewAccount()
|
|
for i := range acc {
|
|
acc[i], addr[i], addrHashes[i] = randomAccount(t)
|
|
if err := blockWriter.UpdateAccountData(ctx, addr[i], &emptyAccount /* original */, acc[i]); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
}
|
|
if err := blockWriter.WriteChangeSets(); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if err := blockWriter.WriteHistory(); 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 = tds.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 != ethdb.ErrKeyNotFound {
|
|
t.Fatal("changeset must be deleted")
|
|
}
|
|
|
|
if debug.IsThinHistory() {
|
|
_, err = db.Get(dbutils.AccountsHistoryBucket, addrHashes[0].Bytes())
|
|
if err != ethdb.ErrKeyNotFound {
|
|
t.Fatal("account must be deleted")
|
|
}
|
|
} else {
|
|
compositeKey, _ := dbutils.CompositeKeySuffix(addrHashes[0].Bytes(), 1)
|
|
_, err = db.Get(dbutils.AccountsHistoryBucket, compositeKey)
|
|
if err != ethdb.ErrKeyNotFound {
|
|
t.Fatal("account must be deleted")
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestMutationCommit(t *testing.T) {
|
|
if debug.IsThinHistory() {
|
|
t.Skip()
|
|
}
|
|
db := ethdb.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(), 2)
|
|
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), 8*(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.Fatal("incorrect storage", i)
|
|
}
|
|
|
|
resAccStorage = make(map[common.Hash]common.Hash)
|
|
err = db.Walk(dbutils.StorageHistoryBucket, dbutils.GenerateStoragePrefix(addrHash, acc.Incarnation), 8*(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(2))
|
|
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(2))
|
|
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)}
|
|
keyHash, err1 := common.HashData(key.Bytes())
|
|
if err1 != nil {
|
|
t.Fatal(err1)
|
|
}
|
|
value := common.Hash{uint8(10 + j)}
|
|
if err2 := expectedChangeSet.Add(dbutils.GenerateCompositeStorageKey(addrHash, accHistory[i].Incarnation, keyHash), value.Bytes()); err2 != nil {
|
|
t.Fatal(err2)
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
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 := ethdb.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), 8*(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.Fatalf("incorrect storage history for %x %x %x", addrHash.String(), v, resultHash)
|
|
}
|
|
}
|
|
}
|
|
|
|
csData, err := db.Get(dbutils.AccountChangeSetBucket, dbutils.EncodeTimestamp(2))
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
expectedChangeSet := changeset.NewAccountChangeSet()
|
|
for i := range addrHashes {
|
|
// Make ajustments for THIN_HISTORY
|
|
c := accHistory[i].SelfCopy()
|
|
copy(c.CodeHash[:], emptyCodeHash)
|
|
c.Root = trie.EmptyRoot
|
|
bLen := c.EncodingLengthForStorage()
|
|
b := make([]byte, bLen)
|
|
c.EncodeForStorage(b)
|
|
innerErr := expectedChangeSet.Add(addrHashes[i].Bytes(), b)
|
|
if innerErr != nil {
|
|
t.Fatal(innerErr)
|
|
}
|
|
}
|
|
sort.Sort(expectedChangeSet)
|
|
expectedData, err := changeset.EncodeAccounts(expectedChangeSet)
|
|
assert.NoError(t, err)
|
|
if !bytes.Equal(csData, expectedData) {
|
|
spew.Dump("res", csData)
|
|
spew.Dump("expected", expectedData)
|
|
t.Fatal("incorrect changeset")
|
|
}
|
|
|
|
csData, err = db.Get(dbutils.StorageChangeSetBucket, dbutils.EncodeTimestamp(2))
|
|
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)}
|
|
keyHash, err1 := common.HashData(key.Bytes())
|
|
if err1 != nil {
|
|
t.Fatal(err1)
|
|
}
|
|
value := common.Hash{uint8(10 + j)}
|
|
if err2 := expectedChangeSet.Add(dbutils.GenerateCompositeStorageKey(addrHash, accHistory[i].Incarnation, keyHash), value.Bytes()); err2 != nil {
|
|
t.Fatal(err2)
|
|
}
|
|
}
|
|
}
|
|
sort.Sort(expectedChangeSet)
|
|
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 ethdb.Database, 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)
|
|
addrs := make([]common.Address, numOfAccounts)
|
|
addrHashes := make([]common.Hash, numOfAccounts)
|
|
tds := NewTrieDbState(common.Hash{}, db, 1)
|
|
blockWriter := tds.DbStateWriter()
|
|
ctx := context.Background()
|
|
for i := range accHistory {
|
|
accHistory[i], addrs[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 + 1)
|
|
|
|
accState[i] = accHistory[i].SelfCopy()
|
|
accState[i].Nonce++
|
|
accState[i].Balance = *big.NewInt(200)
|
|
|
|
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)}
|
|
keyHash, err := common.HashData(key.Bytes())
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
newValue := common.Hash{uint8(j)}
|
|
if newValue != (common.Hash{}) {
|
|
// Empty value is not considered to be present
|
|
accStateStorage[i][keyHash] = newValue
|
|
}
|
|
|
|
value := common.Hash{uint8(10 + j)}
|
|
accHistoryStateStorage[i][keyHash] = value
|
|
if err := blockWriter.WriteAccountStorage(ctx, addrs[i], accHistory[i].Incarnation, &key, &value, &newValue); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
}
|
|
if err := blockWriter.UpdateAccountData(ctx, addrs[i], accHistory[i] /* original */, accState[i]); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
}
|
|
tds.SetBlockNr(2)
|
|
if err := blockWriter.WriteChangeSets(); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if err := blockWriter.WriteHistory(); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
return addrHashes, accState, accStateStorage, accHistory, accHistoryStateStorage
|
|
}
|
|
|
|
func TestMutation_GetAsOf(t *testing.T) {
|
|
db := ethdb.NewMemDatabase()
|
|
mutDB := db.NewBatch()
|
|
tds := NewTrieDbState(common.Hash{}, mutDB, 0)
|
|
blockWriter := tds.DbStateWriter()
|
|
ctx := context.Background()
|
|
emptyAccount := accounts.NewAccount()
|
|
|
|
acc, addr, addrHash := randomAccount(t)
|
|
acc2 := acc.SelfCopy()
|
|
acc2.Nonce = 1
|
|
acc4 := acc.SelfCopy()
|
|
acc4.Nonce = 3
|
|
|
|
tds.SetBlockNr(0)
|
|
if err := blockWriter.UpdateAccountData(ctx, addr, &emptyAccount, acc2); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if err := blockWriter.WriteChangeSets(); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if err := blockWriter.WriteHistory(); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
blockWriter = tds.DbStateWriter()
|
|
tds.SetBlockNr(2)
|
|
if err := blockWriter.UpdateAccountData(ctx, addr, acc2, acc4); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if err := blockWriter.WriteChangeSets(); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if err := blockWriter.WriteHistory(); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
blockWriter = tds.DbStateWriter()
|
|
tds.SetBlockNr(4)
|
|
if err := blockWriter.UpdateAccountData(ctx, addr, acc4, acc); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if err := blockWriter.WriteChangeSets(); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if err := blockWriter.WriteHistory(); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
if _, err := mutDB.Commit(); 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.Address, 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, addr, addrHash
|
|
}
|
|
|
|
func TestBoltDB_WalkAsOf1(t *testing.T) {
|
|
if debug.IsThinHistory() {
|
|
t.Skip()
|
|
}
|
|
db := ethdb.NewMemDatabase()
|
|
tds := NewTrieDbState(common.Hash{}, db, 1)
|
|
blockWriter := tds.DbStateWriter()
|
|
ctx := context.Background()
|
|
emptyVal := common.Hash{}
|
|
|
|
block2Expected := &changeset.ChangeSet{
|
|
Changes: make([]changeset.Change, 0),
|
|
}
|
|
|
|
block4Expected := &changeset.ChangeSet{
|
|
Changes: make([]changeset.Change, 0),
|
|
}
|
|
|
|
block6Expected := &changeset.ChangeSet{
|
|
Changes: make([]changeset.Change, 0),
|
|
}
|
|
|
|
//create state and history
|
|
for i := uint8(1); i <= 7; i++ {
|
|
addr := common.Address{i}
|
|
addrHash, _ := common.HashData(addr[:])
|
|
k := common.Hash{i}
|
|
keyHash, _ := common.HashData(k[:])
|
|
key := dbutils.GenerateCompositeStorageKey(addrHash, 1, keyHash)
|
|
val3 := common.BytesToHash([]byte("block 3 " + strconv.Itoa(int(i))))
|
|
val5 := common.BytesToHash([]byte("block 5 " + strconv.Itoa(int(i))))
|
|
val := common.BytesToHash([]byte("state " + strconv.Itoa(int(i))))
|
|
if i <= 2 {
|
|
if err := blockWriter.WriteAccountStorage(ctx, addr, 1, &k, &val3, &val); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
} else {
|
|
if err := blockWriter.WriteAccountStorage(ctx, addr, 1, &k, &val3, &val5); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
}
|
|
if err := block2Expected.Add(key, []byte("block 3 "+strconv.Itoa(int(i)))); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
}
|
|
tds.SetBlockNr(3)
|
|
if err := blockWriter.WriteChangeSets(); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if err := blockWriter.WriteHistory(); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
blockWriter = tds.DbStateWriter()
|
|
for i := uint8(3); i <= 7; i++ {
|
|
addr := common.Address{i}
|
|
addrHash, _ := common.HashData(addr[:])
|
|
k := common.Hash{i}
|
|
keyHash, _ := common.HashData(k[:])
|
|
key := dbutils.GenerateCompositeStorageKey(addrHash, 1, keyHash)
|
|
val5 := common.BytesToHash([]byte("block 5 " + strconv.Itoa(int(i))))
|
|
val := common.BytesToHash([]byte("state " + strconv.Itoa(int(i))))
|
|
if i > 4 {
|
|
if err := blockWriter.WriteAccountStorage(ctx, addr, 1, &k, &val5, &emptyVal); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
} else {
|
|
if err := blockWriter.WriteAccountStorage(ctx, addr, 1, &k, &val5, &val); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
}
|
|
if err := block4Expected.Add(key, []byte("block 5 "+strconv.Itoa(int(i)))); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
}
|
|
tds.SetBlockNr(5)
|
|
if err := blockWriter.WriteChangeSets(); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if err := blockWriter.WriteHistory(); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
blockWriter = tds.DbStateWriter()
|
|
for i := uint8(1); i < 5; i++ {
|
|
addr := common.Address{i}
|
|
addrHash, _ := common.HashData(addr[:])
|
|
k := common.Hash{i}
|
|
keyHash, _ := common.HashData(k[:])
|
|
key := dbutils.GenerateCompositeStorageKey(addrHash, uint64(1), keyHash)
|
|
val := []byte("state " + strconv.Itoa(int(i)))
|
|
err := block6Expected.Add(key, val)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
if i <= 2 {
|
|
err = block4Expected.Add(key, val)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
}
|
|
}
|
|
tds.SetBlockNr(6)
|
|
if err := blockWriter.WriteChangeSets(); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if err := blockWriter.WriteHistory(); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
block2 := &changeset.ChangeSet{
|
|
Changes: make([]changeset.Change, 0),
|
|
}
|
|
|
|
block4 := &changeset.ChangeSet{
|
|
Changes: make([]changeset.Change, 0),
|
|
}
|
|
|
|
block6 := &changeset.ChangeSet{
|
|
Changes: make([]changeset.Change, 0),
|
|
}
|
|
|
|
//walk and collect walkAsOf result
|
|
var err error
|
|
var startKey [72]byte
|
|
err = db.WalkAsOf(dbutils.StorageBucket, dbutils.StorageHistoryBucket, startKey[:], 0, 2, func(k []byte, v []byte) (b bool, e error) {
|
|
err = block2.Add(common.CopyBytes(k), common.CopyBytes(v))
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
return true, nil
|
|
})
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
err = db.WalkAsOf(dbutils.StorageBucket, dbutils.StorageHistoryBucket, startKey[:], 0, 4, func(k []byte, v []byte) (b bool, e error) {
|
|
err = block4.Add(common.CopyBytes(k), common.CopyBytes(v))
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
return true, nil
|
|
})
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
err = db.WalkAsOf(dbutils.StorageBucket, dbutils.StorageHistoryBucket, startKey[:], 0, 6, func(k []byte, v []byte) (b bool, e error) {
|
|
err = block6.Add(common.CopyBytes(k), common.CopyBytes(v))
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
return true, nil
|
|
})
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
sort.Sort(block2Expected)
|
|
if !reflect.DeepEqual(block2, block2Expected) {
|
|
spew.Dump("expected", block2Expected)
|
|
spew.Dump("current", block2)
|
|
t.Fatal("block 2 result is incorrect")
|
|
}
|
|
sort.Sort(block4Expected)
|
|
if !reflect.DeepEqual(block4, block4Expected) {
|
|
spew.Dump("expected", block4Expected)
|
|
spew.Dump("current", block4)
|
|
t.Fatal("block 4 result is incorrect")
|
|
}
|
|
sort.Sort(block6Expected)
|
|
if !reflect.DeepEqual(block6, block6Expected) {
|
|
spew.Dump("expected", block6Expected)
|
|
spew.Dump("current", block6)
|
|
t.Fatal("block 6 result is incorrect")
|
|
}
|
|
}
|