mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-06 19:12:19 +00:00
436493350e
1. changes sentinel to use an http-like interface 2. moves hexutil, crypto/blake2b, metrics packages to erigon-lib
1148 lines
28 KiB
Go
1148 lines
28 KiB
Go
package state
|
|
|
|
import (
|
|
"fmt"
|
|
"github.com/ledgerwatch/erigon-lib/kv/dbutils"
|
|
"math/rand"
|
|
"reflect"
|
|
"sort"
|
|
"strconv"
|
|
"testing"
|
|
|
|
"github.com/davecgh/go-spew/spew"
|
|
"github.com/holiman/uint256"
|
|
libcommon "github.com/ledgerwatch/erigon-lib/common"
|
|
"github.com/ledgerwatch/erigon-lib/common/length"
|
|
"github.com/ledgerwatch/erigon-lib/kv"
|
|
"github.com/ledgerwatch/erigon-lib/kv/bitmapdb"
|
|
"github.com/ledgerwatch/erigon-lib/kv/memdb"
|
|
"github.com/ledgerwatch/erigon-lib/kv/temporal/historyv2"
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
|
|
"github.com/ledgerwatch/erigon/common/math"
|
|
"github.com/ledgerwatch/erigon/core/state/historyv2read"
|
|
"github.com/ledgerwatch/erigon/core/types/accounts"
|
|
"github.com/ledgerwatch/erigon/crypto"
|
|
"github.com/ledgerwatch/erigon/turbo/trie"
|
|
)
|
|
|
|
func TestMutationDeleteTimestamp(t *testing.T) {
|
|
_, tx := memdb.NewTestTx(t)
|
|
|
|
acc := make([]*accounts.Account, 10)
|
|
addr := make([]libcommon.Address, 10)
|
|
blockWriter := NewPlainStateWriter(tx, tx, 1)
|
|
emptyAccount := accounts.NewAccount()
|
|
for i := range acc {
|
|
acc[i], addr[i] = randomAccount(t)
|
|
if err := blockWriter.UpdateAccountData(addr[i], &emptyAccount, 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)
|
|
}
|
|
|
|
i := 0
|
|
err := historyv2.ForEach(tx, kv.AccountChangeSet, nil, func(blockN uint64, k, v []byte) error {
|
|
i++
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if i != 10 {
|
|
t.FailNow()
|
|
}
|
|
|
|
index, err := bitmapdb.Get64(tx, kv.E2AccountsHistory, addr[0].Bytes(), 0, math.MaxUint32)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
parsed := index.ToArray()
|
|
if parsed[0] != 1 {
|
|
t.Fatal("incorrect block num")
|
|
}
|
|
|
|
count := 0
|
|
err = historyv2.ForPrefix(tx, kv.StorageChangeSet, dbutils.EncodeBlockNumber(1), func(blockN uint64, k, v []byte) error {
|
|
count++
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if count != 0 {
|
|
t.Fatal("changeset must be deleted")
|
|
}
|
|
|
|
found, err := tx.GetOne(kv.E2AccountsHistory, addr[0].Bytes())
|
|
require.NoError(t, err)
|
|
require.Nil(t, found, "account must be deleted")
|
|
}
|
|
|
|
func TestMutationCommit(t *testing.T) {
|
|
_, tx := memdb.NewTestTx(t)
|
|
|
|
numOfAccounts := 5
|
|
numOfStateKeys := 5
|
|
|
|
addrs, accState, accStateStorage, accHistory, accHistoryStateStorage := generateAccountsWithStorageAndHistory(t, NewPlainStateWriter(tx, tx, 2), numOfAccounts, numOfStateKeys)
|
|
|
|
for i, addr := range addrs {
|
|
acc, err := NewPlainStateReader(tx).ReadAccountData(addr)
|
|
|
|
if err != nil {
|
|
t.Fatal("error on get account", i, err)
|
|
} else if acc == nil {
|
|
t.Fatal("error on get account", i)
|
|
}
|
|
|
|
if !accState[i].Equals(acc) {
|
|
spew.Dump("got", acc)
|
|
spew.Dump("expected", accState[i])
|
|
t.Fatal("Accounts not equals")
|
|
}
|
|
|
|
index, err := bitmapdb.Get64(tx, kv.E2AccountsHistory, addr.Bytes(), 0, math.MaxUint32)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
parsedIndex := index.ToArray()
|
|
if parsedIndex[0] != 2 || index.GetCardinality() != 1 {
|
|
t.Fatal("incorrect history index")
|
|
}
|
|
|
|
resAccStorage := make(map[libcommon.Hash]uint256.Int)
|
|
err = tx.ForPrefix(kv.PlainState, dbutils.PlainGenerateStoragePrefix(addr[:], acc.Incarnation), func(k, v []byte) error {
|
|
resAccStorage[libcommon.BytesToHash(k[length.Addr+8:])] = *uint256.NewInt(0).SetBytes(v)
|
|
return 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] {
|
|
c1, _ := tx.Cursor(kv.E2StorageHistory)
|
|
c2, _ := tx.CursorDupSort(kv.StorageChangeSet)
|
|
res, _, err := historyv2read.GetAsOf(tx, c1, c2, true /* storage */, dbutils.PlainGenerateCompositeStorageKey(addr.Bytes(), acc.Incarnation, k.Bytes()), 1)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
result := uint256.NewInt(0).SetBytes(res)
|
|
if !v.Eq(result) {
|
|
t.Fatalf("incorrect storage history for %x %x %x", addr.String(), v, result)
|
|
}
|
|
}
|
|
}
|
|
|
|
changeSetInDB := historyv2.NewAccountChangeSet()
|
|
err := historyv2.ForPrefix(tx, kv.AccountChangeSet, dbutils.EncodeBlockNumber(2), func(_ uint64, k, v []byte) error {
|
|
if err := changeSetInDB.Add(k, v); err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
expectedChangeSet := historyv2.NewAccountChangeSet()
|
|
for i := range addrs {
|
|
// 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(addrs[i].Bytes(), b)
|
|
if innerErr != nil {
|
|
t.Fatal(innerErr)
|
|
}
|
|
}
|
|
sort.Sort(expectedChangeSet)
|
|
if !reflect.DeepEqual(changeSetInDB, expectedChangeSet) {
|
|
spew.Dump("res", changeSetInDB)
|
|
spew.Dump("expected", expectedChangeSet)
|
|
t.Fatal("incorrect changeset")
|
|
}
|
|
|
|
cs := historyv2.NewStorageChangeSet()
|
|
err = historyv2.ForPrefix(tx, kv.StorageChangeSet, dbutils.EncodeBlockNumber(2), func(_ uint64, k, v []byte) error {
|
|
if err2 := cs.Add(k, v); err2 != nil {
|
|
return err2
|
|
}
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
if cs.Len() != numOfAccounts*numOfStateKeys {
|
|
t.Errorf("Length does not match, got %d, expected %d", cs.Len(), numOfAccounts*numOfStateKeys)
|
|
}
|
|
|
|
expectedChangeSet = historyv2.NewStorageChangeSet()
|
|
for i, addr := range addrs {
|
|
for j := 0; j < numOfStateKeys; j++ {
|
|
key := libcommon.Hash{uint8(i*100 + j)}
|
|
value := uint256.NewInt(uint64(10 + j))
|
|
if err2 := expectedChangeSet.Add(dbutils.PlainGenerateCompositeStorageKey(addr.Bytes(), accHistory[i].Incarnation, key.Bytes()), value.Bytes()); err2 != nil {
|
|
t.Fatal(err2)
|
|
}
|
|
}
|
|
}
|
|
sort.Sort(expectedChangeSet)
|
|
|
|
assert.Equal(t, cs, expectedChangeSet)
|
|
}
|
|
|
|
func generateAccountsWithStorageAndHistory(t *testing.T, blockWriter *PlainStateWriter, numOfAccounts, numOfStateKeys int) ([]libcommon.Address, []*accounts.Account, []map[libcommon.Hash]uint256.Int, []*accounts.Account, []map[libcommon.Hash]uint256.Int) {
|
|
t.Helper()
|
|
|
|
accHistory := make([]*accounts.Account, numOfAccounts)
|
|
accState := make([]*accounts.Account, numOfAccounts)
|
|
accStateStorage := make([]map[libcommon.Hash]uint256.Int, numOfAccounts)
|
|
accHistoryStateStorage := make([]map[libcommon.Hash]uint256.Int, numOfAccounts)
|
|
addrs := make([]libcommon.Address, numOfAccounts)
|
|
for i := range accHistory {
|
|
accHistory[i], addrs[i] = randomAccount(t)
|
|
accHistory[i].Balance = *uint256.NewInt(100)
|
|
accHistory[i].CodeHash = libcommon.Hash{uint8(10 + i)}
|
|
accHistory[i].Root = libcommon.Hash{uint8(10 + i)}
|
|
accHistory[i].Incarnation = uint64(i + 1)
|
|
|
|
accState[i] = accHistory[i].SelfCopy()
|
|
accState[i].Nonce++
|
|
accState[i].Balance = *uint256.NewInt(200)
|
|
|
|
accStateStorage[i] = make(map[libcommon.Hash]uint256.Int)
|
|
accHistoryStateStorage[i] = make(map[libcommon.Hash]uint256.Int)
|
|
for j := 0; j < numOfStateKeys; j++ {
|
|
key := libcommon.Hash{uint8(i*100 + j)}
|
|
newValue := uint256.NewInt(uint64(j))
|
|
if !newValue.IsZero() {
|
|
// Empty value is not considered to be present
|
|
accStateStorage[i][key] = *newValue
|
|
}
|
|
|
|
value := uint256.NewInt(uint64(10 + j))
|
|
accHistoryStateStorage[i][key] = *value
|
|
if err := blockWriter.WriteAccountStorage(addrs[i], accHistory[i].Incarnation, &key, value, newValue); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
}
|
|
if err := blockWriter.UpdateAccountData(addrs[i], accHistory[i], accState[i]); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
}
|
|
if err := blockWriter.WriteChangeSets(); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if err := blockWriter.WriteHistory(); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
return addrs, accState, accStateStorage, accHistory, accHistoryStateStorage
|
|
}
|
|
|
|
func randomAccount(t *testing.T) (*accounts.Account, libcommon.Address) {
|
|
t.Helper()
|
|
key, err := crypto.GenerateKey()
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
acc := accounts.NewAccount()
|
|
acc.Initialised = true
|
|
acc.Balance = *uint256.NewInt(uint64(rand.Int63()))
|
|
addr := crypto.PubkeyToAddress(key.PublicKey)
|
|
return &acc, addr
|
|
}
|
|
|
|
/*
|
|
before 3:
|
|
addr1(f22b):""
|
|
addr2(1f0e):""
|
|
addr3(3e05):""
|
|
addr4(d12e):""
|
|
block 3
|
|
addr1(f22b):"block3"
|
|
addr2(1f0e):""
|
|
addr3(3e05):"state"
|
|
addr4(d12e):"block3"
|
|
block 5
|
|
addr1(f22b):"state"
|
|
addr2(1f0e):"state"
|
|
addr3(3e05):"state"
|
|
addr4(d12e):""
|
|
*/
|
|
|
|
func TestWalkAsOfStatePlain(t *testing.T) {
|
|
_, tx := memdb.NewTestTx(t)
|
|
|
|
emptyVal := uint256.NewInt(0)
|
|
block3Val := uint256.NewInt(0).SetBytes([]byte("block 3"))
|
|
stateVal := uint256.NewInt(0).SetBytes([]byte("state"))
|
|
numOfAccounts := uint8(4)
|
|
addrs := make([]libcommon.Address, numOfAccounts)
|
|
key := libcommon.Hash{123}
|
|
for i := uint8(0); i < numOfAccounts; i++ {
|
|
addrs[i] = libcommon.Address{i + 1}
|
|
}
|
|
|
|
block2Expected := &historyv2.ChangeSet{
|
|
Changes: make([]historyv2.Change, 0),
|
|
}
|
|
|
|
block4Expected := &historyv2.ChangeSet{
|
|
Changes: make([]historyv2.Change, 0),
|
|
}
|
|
|
|
block6Expected := &historyv2.ChangeSet{
|
|
Changes: make([]historyv2.Change, 0),
|
|
}
|
|
|
|
withoutInc := func(addr libcommon.Address, keyHash libcommon.Hash) []byte {
|
|
expectedKey := make([]byte, length.Hash+length.Addr)
|
|
copy(expectedKey[:length.Addr], addr.Bytes())
|
|
copy(expectedKey[length.Addr:], keyHash.Bytes())
|
|
return expectedKey
|
|
}
|
|
|
|
writeStorageBlockData(t, NewPlainStateWriter(tx, tx, 3), []storageData{
|
|
{
|
|
addrs[0],
|
|
historyv2read.DefaultIncarnation,
|
|
key,
|
|
emptyVal,
|
|
block3Val,
|
|
},
|
|
{
|
|
addrs[2],
|
|
historyv2read.DefaultIncarnation,
|
|
key,
|
|
emptyVal,
|
|
stateVal,
|
|
},
|
|
{
|
|
addrs[3],
|
|
historyv2read.DefaultIncarnation,
|
|
key,
|
|
emptyVal,
|
|
block3Val,
|
|
},
|
|
})
|
|
|
|
writeStorageBlockData(t, NewPlainStateWriter(tx, tx, 5), []storageData{
|
|
{
|
|
addrs[0],
|
|
historyv2read.DefaultIncarnation,
|
|
key,
|
|
block3Val,
|
|
stateVal,
|
|
},
|
|
{
|
|
addrs[1],
|
|
historyv2read.DefaultIncarnation,
|
|
key,
|
|
emptyVal,
|
|
stateVal,
|
|
},
|
|
{
|
|
addrs[3],
|
|
historyv2read.DefaultIncarnation,
|
|
key,
|
|
block3Val,
|
|
emptyVal,
|
|
},
|
|
})
|
|
|
|
block2 := &historyv2.ChangeSet{
|
|
Changes: make([]historyv2.Change, 0),
|
|
}
|
|
|
|
for _, addr := range addrs {
|
|
if err := WalkAsOfStorage(tx, addr, historyv2read.DefaultIncarnation, libcommon.Hash{}, 2, func(kAddr, kLoc []byte, v []byte) (b bool, e error) {
|
|
err := block2.Add(append(libcommon.CopyBytes(kAddr), kLoc...), libcommon.CopyBytes(v))
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
return true, nil
|
|
}); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
}
|
|
assertChangesEquals(t, block2, block2Expected)
|
|
|
|
block4 := &historyv2.ChangeSet{
|
|
Changes: make([]historyv2.Change, 0),
|
|
}
|
|
for _, addr := range addrs {
|
|
if err := WalkAsOfStorage(tx, addr, historyv2read.DefaultIncarnation, libcommon.Hash{}, 4, func(kAddr, kLoc []byte, v []byte) (b bool, e error) {
|
|
err := block4.Add(append(libcommon.CopyBytes(kAddr), kLoc...), libcommon.CopyBytes(v))
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
return true, nil
|
|
}); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
}
|
|
|
|
block4Expected.Changes = []historyv2.Change{
|
|
{
|
|
Key: withoutInc(addrs[0], key),
|
|
Value: block3Val.Bytes(),
|
|
},
|
|
{
|
|
Key: withoutInc(addrs[2], key),
|
|
Value: stateVal.Bytes(),
|
|
},
|
|
{
|
|
Key: withoutInc(addrs[3], key),
|
|
Value: block3Val.Bytes(),
|
|
},
|
|
}
|
|
assertChangesEquals(t, block4, block4Expected)
|
|
|
|
block6 := &historyv2.ChangeSet{
|
|
Changes: make([]historyv2.Change, 0),
|
|
}
|
|
for _, addr := range addrs {
|
|
if err := WalkAsOfStorage(tx, addr, historyv2read.DefaultIncarnation, libcommon.Hash{}, 6, func(kAddr, kLoc []byte, v []byte) (b bool, e error) {
|
|
err := block6.Add(append(libcommon.CopyBytes(kAddr), kLoc...), libcommon.CopyBytes(v))
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
return true, nil
|
|
}); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
}
|
|
|
|
block6Expected.Changes = []historyv2.Change{
|
|
{
|
|
Key: withoutInc(addrs[0], key),
|
|
Value: stateVal.Bytes(),
|
|
},
|
|
{
|
|
Key: withoutInc(addrs[1], key),
|
|
Value: stateVal.Bytes(),
|
|
},
|
|
{
|
|
Key: withoutInc(addrs[2], key),
|
|
Value: stateVal.Bytes(),
|
|
},
|
|
}
|
|
assertChangesEquals(t, block6, block6Expected)
|
|
}
|
|
|
|
func TestWalkAsOfUsingFixedBytesStatePlain(t *testing.T) {
|
|
_, tx := memdb.NewTestTx(t)
|
|
|
|
emptyVal := uint256.NewInt(0)
|
|
block3Val := uint256.NewInt(0).SetBytes([]byte("block 3"))
|
|
stateVal := uint256.NewInt(0).SetBytes([]byte("state"))
|
|
|
|
addr1 := libcommon.Address{1}
|
|
addr2 := libcommon.Address{2}
|
|
|
|
key1 := libcommon.Hash{1}
|
|
key2 := libcommon.Hash{2}
|
|
key3 := libcommon.Hash{3}
|
|
|
|
block2Expected := &historyv2.ChangeSet{
|
|
Changes: make([]historyv2.Change, 0),
|
|
}
|
|
|
|
block4Expected := &historyv2.ChangeSet{
|
|
Changes: make([]historyv2.Change, 0),
|
|
}
|
|
|
|
block6Expected := &historyv2.ChangeSet{
|
|
Changes: make([]historyv2.Change, 0),
|
|
}
|
|
|
|
withoutInc := func(addr libcommon.Address, keyHash libcommon.Hash) []byte {
|
|
expectedKey := make([]byte, length.Hash+length.Addr)
|
|
copy(expectedKey[:length.Addr], addr.Bytes())
|
|
copy(expectedKey[length.Addr:], keyHash.Bytes())
|
|
return expectedKey
|
|
}
|
|
|
|
writeStorageBlockData(t, NewPlainStateWriter(tx, tx, 3), []storageData{
|
|
{
|
|
addr: addr1,
|
|
inc: historyv2read.DefaultIncarnation,
|
|
key: key1,
|
|
oldVal: emptyVal,
|
|
newVal: block3Val,
|
|
},
|
|
{
|
|
addr: addr1,
|
|
inc: historyv2read.DefaultIncarnation,
|
|
key: key2,
|
|
oldVal: emptyVal,
|
|
newVal: block3Val,
|
|
},
|
|
{
|
|
addr: addr1,
|
|
inc: historyv2read.DefaultIncarnation,
|
|
key: key3,
|
|
oldVal: emptyVal,
|
|
newVal: block3Val,
|
|
},
|
|
{
|
|
addr: addr2,
|
|
inc: historyv2read.DefaultIncarnation,
|
|
key: key3,
|
|
oldVal: emptyVal,
|
|
newVal: block3Val,
|
|
},
|
|
})
|
|
|
|
writeStorageBlockData(t, NewPlainStateWriter(tx, tx, 5), []storageData{
|
|
{
|
|
addr: addr1,
|
|
inc: historyv2read.DefaultIncarnation,
|
|
key: key1,
|
|
oldVal: block3Val,
|
|
newVal: stateVal,
|
|
},
|
|
{
|
|
addr: addr1,
|
|
inc: historyv2read.DefaultIncarnation,
|
|
key: key2,
|
|
oldVal: block3Val,
|
|
newVal: stateVal,
|
|
},
|
|
{
|
|
addr: addr1,
|
|
inc: historyv2read.DefaultIncarnation,
|
|
key: key3,
|
|
oldVal: block3Val,
|
|
newVal: emptyVal,
|
|
},
|
|
{
|
|
addr: addr2,
|
|
inc: historyv2read.DefaultIncarnation,
|
|
key: key3,
|
|
oldVal: block3Val,
|
|
newVal: stateVal,
|
|
},
|
|
})
|
|
|
|
block2 := &historyv2.ChangeSet{
|
|
Changes: make([]historyv2.Change, 0),
|
|
}
|
|
|
|
//walk and collect walkAsOf result
|
|
startKey := make([]byte, 60)
|
|
copy(startKey[:length.Addr], addr1.Bytes())
|
|
|
|
if err := WalkAsOfStorage(tx, addr1, historyv2read.DefaultIncarnation, libcommon.Hash{}, 2, func(kAddr, kLoc []byte, v []byte) (b bool, e error) {
|
|
err := block2.Add(append(libcommon.CopyBytes(kAddr), kLoc...), libcommon.CopyBytes(v))
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
return true, nil
|
|
}); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
assertChangesEquals(t, block2, block2Expected)
|
|
|
|
block4 := &historyv2.ChangeSet{
|
|
Changes: make([]historyv2.Change, 0),
|
|
}
|
|
if err := WalkAsOfStorage(tx, addr1, historyv2read.DefaultIncarnation, libcommon.Hash{}, 4, func(kAddr, kLoc []byte, v []byte) (b bool, e error) {
|
|
err := block4.Add(append(libcommon.CopyBytes(kAddr), kLoc...), libcommon.CopyBytes(v))
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
return true, nil
|
|
}); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
block4Expected.Changes = []historyv2.Change{
|
|
{
|
|
Key: withoutInc(addr1, key1),
|
|
Value: block3Val.Bytes(),
|
|
},
|
|
{
|
|
Key: withoutInc(addr1, key2),
|
|
Value: block3Val.Bytes(),
|
|
},
|
|
{
|
|
Key: withoutInc(addr1, key3),
|
|
Value: block3Val.Bytes(),
|
|
},
|
|
}
|
|
assertChangesEquals(t, block4, block4Expected)
|
|
|
|
block4.Changes = block4.Changes[:0]
|
|
for _, addr := range []libcommon.Address{addr1, addr2} {
|
|
if err := WalkAsOfStorage(tx, addr, historyv2read.DefaultIncarnation, libcommon.Hash{}, 4, func(kAddr, kLoc []byte, v []byte) (b bool, e error) {
|
|
err := block4.Add(append(libcommon.CopyBytes(kAddr), kLoc...), libcommon.CopyBytes(v))
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
return true, nil
|
|
}); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
}
|
|
|
|
block4Expected.Changes = append(block4Expected.Changes, historyv2.Change{
|
|
Key: withoutInc(addr2, key3),
|
|
Value: block3Val.Bytes(),
|
|
})
|
|
assertChangesEquals(t, block4, block4Expected)
|
|
|
|
block6 := &historyv2.ChangeSet{
|
|
Changes: make([]historyv2.Change, 0),
|
|
}
|
|
if err := WalkAsOfStorage(tx, addr1, historyv2read.DefaultIncarnation, libcommon.Hash{}, 6, func(kAddr, kLoc []byte, v []byte) (b bool, e error) {
|
|
err := block6.Add(append(libcommon.CopyBytes(kAddr), kLoc...), libcommon.CopyBytes(v))
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
return true, nil
|
|
}); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
block6Expected.Changes = []historyv2.Change{
|
|
{
|
|
Key: withoutInc(addr1, key1),
|
|
Value: stateVal.Bytes(),
|
|
},
|
|
{
|
|
Key: withoutInc(addr1, key2),
|
|
Value: stateVal.Bytes(),
|
|
},
|
|
}
|
|
assertChangesEquals(t, block6, block6Expected)
|
|
|
|
block6.Changes = block6.Changes[:0]
|
|
for _, addr := range []libcommon.Address{addr1, addr2} {
|
|
if err := WalkAsOfStorage(tx, addr, historyv2read.DefaultIncarnation, libcommon.Hash{}, 6, func(kAddr, kLoc []byte, v []byte) (b bool, e error) {
|
|
err := block6.Add(append(libcommon.CopyBytes(kAddr), kLoc...), libcommon.CopyBytes(v))
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
return true, nil
|
|
}); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
}
|
|
block6Expected.Changes = append(block6Expected.Changes, historyv2.Change{
|
|
Key: withoutInc(addr2, key3),
|
|
Value: stateVal.Bytes(),
|
|
})
|
|
assertChangesEquals(t, block6, block6Expected)
|
|
}
|
|
|
|
func TestWalkAsOfAccountPlain(t *testing.T) {
|
|
_, tx := memdb.NewTestTx(t)
|
|
|
|
emptyValAcc := accounts.NewAccount()
|
|
emptyVal := make([]byte, emptyValAcc.EncodingLengthForStorage())
|
|
emptyValAcc.EncodeForStorage(emptyVal)
|
|
|
|
block3ValAcc := emptyValAcc.SelfCopy()
|
|
block3ValAcc.Nonce = 3
|
|
block3ValAcc.Initialised = true
|
|
block3Val := make([]byte, block3ValAcc.EncodingLengthForStorage())
|
|
block3ValAcc.EncodeForStorage(block3Val)
|
|
|
|
stateValAcc := emptyValAcc.SelfCopy()
|
|
stateValAcc.Nonce = 5
|
|
stateValAcc.Initialised = true
|
|
stateVal := make([]byte, stateValAcc.EncodingLengthForStorage())
|
|
stateValAcc.EncodeForStorage(stateVal)
|
|
|
|
numOfAccounts := uint8(4)
|
|
addrs := make([]libcommon.Address, numOfAccounts)
|
|
addrHashes := make([]libcommon.Hash, numOfAccounts)
|
|
for i := uint8(0); i < numOfAccounts; i++ {
|
|
addrs[i] = libcommon.Address{i + 1}
|
|
addrHash, _ := libcommon.HashData(addrs[i].Bytes())
|
|
addrHashes[i] = addrHash
|
|
}
|
|
|
|
block2 := &historyv2.ChangeSet{
|
|
Changes: make([]historyv2.Change, 0),
|
|
}
|
|
|
|
block2Expected := &historyv2.ChangeSet{
|
|
Changes: make([]historyv2.Change, 0),
|
|
}
|
|
|
|
writeBlockData(t, NewPlainStateWriter(tx, tx, 3), []accData{
|
|
{
|
|
addr: addrs[0],
|
|
oldVal: &emptyValAcc,
|
|
newVal: block3ValAcc,
|
|
},
|
|
{
|
|
addr: addrs[2],
|
|
oldVal: &emptyValAcc,
|
|
newVal: block3ValAcc,
|
|
},
|
|
{
|
|
addr: addrs[3],
|
|
oldVal: &emptyValAcc,
|
|
newVal: block3ValAcc,
|
|
},
|
|
})
|
|
|
|
writeBlockData(t, NewPlainStateWriter(tx, tx, 5), []accData{
|
|
{
|
|
addr: addrs[0],
|
|
oldVal: block3ValAcc,
|
|
newVal: stateValAcc,
|
|
},
|
|
{
|
|
addr: addrs[1],
|
|
oldVal: &emptyValAcc,
|
|
newVal: stateValAcc,
|
|
},
|
|
{
|
|
addr: addrs[3],
|
|
oldVal: block3ValAcc,
|
|
newVal: nil,
|
|
},
|
|
})
|
|
|
|
if err := WalkAsOfAccounts(tx, libcommon.Address{}, 2, func(k []byte, v []byte) (b bool, e error) {
|
|
innerErr := block2.Add(libcommon.CopyBytes(k), libcommon.CopyBytes(v))
|
|
if innerErr != nil {
|
|
t.Fatal(innerErr)
|
|
}
|
|
return true, nil
|
|
}); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
assertChangesEquals(t, block2, block2Expected)
|
|
|
|
block4 := &historyv2.ChangeSet{
|
|
Changes: make([]historyv2.Change, 0),
|
|
}
|
|
|
|
block4Expected := &historyv2.ChangeSet{
|
|
Changes: []historyv2.Change{
|
|
{
|
|
Key: addrs[0].Bytes(),
|
|
Value: block3Val,
|
|
},
|
|
{
|
|
Key: addrs[2].Bytes(),
|
|
Value: block3Val,
|
|
},
|
|
{
|
|
Key: addrs[3].Bytes(),
|
|
Value: block3Val,
|
|
},
|
|
},
|
|
}
|
|
|
|
if err := WalkAsOfAccounts(tx, libcommon.Address{}, 4, func(k []byte, v []byte) (b bool, e error) {
|
|
innerErr := block4.Add(libcommon.CopyBytes(k), libcommon.CopyBytes(v))
|
|
if innerErr != nil {
|
|
t.Fatal(innerErr)
|
|
}
|
|
return true, nil
|
|
}); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
assertChangesEquals(t, block4, block4Expected)
|
|
|
|
block6 := &historyv2.ChangeSet{
|
|
Changes: make([]historyv2.Change, 0),
|
|
}
|
|
|
|
block6Expected := &historyv2.ChangeSet{
|
|
Changes: []historyv2.Change{
|
|
{
|
|
Key: addrs[0].Bytes(),
|
|
Value: stateVal,
|
|
},
|
|
{
|
|
Key: addrs[1].Bytes(),
|
|
Value: stateVal,
|
|
},
|
|
{
|
|
Key: addrs[2].Bytes(),
|
|
Value: block3Val,
|
|
},
|
|
},
|
|
}
|
|
|
|
if err := WalkAsOfAccounts(tx, libcommon.Address{}, 6, func(k []byte, v []byte) (b bool, e error) {
|
|
innerErr := block6.Add(libcommon.CopyBytes(k), libcommon.CopyBytes(v))
|
|
if innerErr != nil {
|
|
t.Fatal(innerErr)
|
|
}
|
|
return true, nil
|
|
}); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
assertChangesEquals(t, block6, block6Expected)
|
|
}
|
|
|
|
func TestWalkAsOfAccountPlain_WithChunks(t *testing.T) {
|
|
_, tx := memdb.NewTestTx(t)
|
|
|
|
emptyValAcc := accounts.NewAccount()
|
|
emptyVal := make([]byte, emptyValAcc.EncodingLengthForStorage())
|
|
emptyValAcc.EncodeForStorage(emptyVal)
|
|
|
|
block3ValAcc := emptyValAcc.SelfCopy()
|
|
block3ValAcc.Nonce = 3
|
|
block3ValAcc.Initialised = true
|
|
block3Val := make([]byte, block3ValAcc.EncodingLengthForStorage())
|
|
block3ValAcc.EncodeForStorage(block3Val)
|
|
|
|
stateValAcc := emptyValAcc.SelfCopy()
|
|
stateValAcc.Nonce = 5
|
|
stateValAcc.Initialised = true
|
|
stateVal := make([]byte, stateValAcc.EncodingLengthForStorage())
|
|
stateValAcc.EncodeForStorage(stateVal)
|
|
|
|
numOfAccounts := uint8(4)
|
|
addrs := make([]libcommon.Address, numOfAccounts)
|
|
addrHashes := make([]libcommon.Hash, numOfAccounts)
|
|
for i := uint8(0); i < numOfAccounts; i++ {
|
|
addrs[i] = libcommon.Address{i + 1}
|
|
addrHash, _ := libcommon.HashData(addrs[i].Bytes())
|
|
addrHashes[i] = addrHash
|
|
}
|
|
|
|
addr1Old := emptyValAcc.SelfCopy()
|
|
addr1Old.Initialised = true
|
|
addr1Old.Nonce = 1
|
|
addr2Old := emptyValAcc.SelfCopy()
|
|
addr2Old.Initialised = true
|
|
addr2Old.Nonce = 1
|
|
addr3Old := emptyValAcc.SelfCopy()
|
|
addr3Old.Initialised = true
|
|
addr3Old.Nonce = 1
|
|
|
|
var addr1New, addr2New, addr3New *accounts.Account
|
|
|
|
writeBlockData(t, NewPlainStateWriter(tx, tx, 1), []accData{
|
|
{
|
|
addr: addrs[0],
|
|
oldVal: &emptyValAcc,
|
|
newVal: addr1Old,
|
|
},
|
|
{
|
|
addr: addrs[1],
|
|
oldVal: &emptyValAcc,
|
|
newVal: addr1Old,
|
|
},
|
|
{
|
|
addr: addrs[2],
|
|
oldVal: &emptyValAcc,
|
|
newVal: addr1Old,
|
|
},
|
|
})
|
|
|
|
for i := 2; i < 1100; i++ {
|
|
addr1New = addr1Old.SelfCopy()
|
|
addr1New.Nonce = uint64(i)
|
|
addr2New = addr2Old.SelfCopy()
|
|
addr2New.Nonce = uint64(i)
|
|
addr3New = addr3Old.SelfCopy()
|
|
addr3New.Nonce = uint64(i)
|
|
writeBlockData(t, NewPlainStateWriter(tx, tx, uint64(i)), []accData{
|
|
{
|
|
addr: addrs[0],
|
|
oldVal: addr1Old,
|
|
newVal: addr1New,
|
|
},
|
|
{
|
|
addr: addrs[1],
|
|
oldVal: addr2Old,
|
|
newVal: addr2New,
|
|
},
|
|
{
|
|
addr: addrs[2],
|
|
oldVal: addr3Old,
|
|
newVal: addr3New,
|
|
},
|
|
})
|
|
addr1Old = addr1New.SelfCopy()
|
|
addr2Old = addr2New.SelfCopy()
|
|
addr3Old = addr3New.SelfCopy()
|
|
}
|
|
|
|
addr1New = addr1Old.SelfCopy()
|
|
addr1New.Nonce = 1100
|
|
addr2New = addr2Old.SelfCopy()
|
|
addr2New.Nonce = 1100
|
|
addr3New = addr3Old.SelfCopy()
|
|
addr3New.Nonce = 1100
|
|
|
|
writeBlockData(t, NewPlainStateWriter(tx, tx, 1100), []accData{
|
|
{
|
|
addr: addrs[0],
|
|
oldVal: addr1Old,
|
|
newVal: addr1New,
|
|
},
|
|
{
|
|
addr: addrs[1],
|
|
oldVal: addr1Old,
|
|
newVal: addr1New,
|
|
},
|
|
{
|
|
addr: addrs[2],
|
|
oldVal: addr1Old,
|
|
newVal: addr1New,
|
|
},
|
|
})
|
|
|
|
for _, blockNum := range []uint64{5, 100, 1000, 1050} {
|
|
obtained := &historyv2.ChangeSet{
|
|
Changes: make([]historyv2.Change, 0),
|
|
}
|
|
|
|
if err := WalkAsOfAccounts(tx, libcommon.Address{}, blockNum, func(k []byte, v []byte) (b bool, e error) {
|
|
innerErr := obtained.Add(libcommon.CopyBytes(k), libcommon.CopyBytes(v))
|
|
if innerErr != nil {
|
|
t.Fatal(innerErr)
|
|
}
|
|
return true, nil
|
|
}); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
acc := addr1Old.SelfCopy()
|
|
acc.Nonce = blockNum - 1
|
|
accBytes := make([]byte, acc.EncodingLengthForStorage())
|
|
acc.EncodeForStorage(accBytes)
|
|
expected := &historyv2.ChangeSet{
|
|
Changes: []historyv2.Change{
|
|
{
|
|
Key: addrs[0].Bytes(),
|
|
Value: accBytes,
|
|
},
|
|
{
|
|
Key: addrs[1].Bytes(),
|
|
Value: accBytes,
|
|
},
|
|
{
|
|
Key: addrs[2].Bytes(),
|
|
Value: accBytes,
|
|
},
|
|
},
|
|
}
|
|
assertChangesEquals(t, obtained, expected)
|
|
}
|
|
}
|
|
|
|
func TestWalkAsOfStoragePlain_WithChunks(t *testing.T) {
|
|
_, tx := memdb.NewTestTx(t)
|
|
|
|
numOfAccounts := uint8(4)
|
|
addrs := make([]libcommon.Address, numOfAccounts)
|
|
addrHashes := make([]libcommon.Hash, numOfAccounts)
|
|
for i := uint8(0); i < numOfAccounts; i++ {
|
|
addrs[i] = libcommon.Address{i + 1}
|
|
addrHash, _ := libcommon.HashData(addrs[i].Bytes())
|
|
addrHashes[i] = addrHash
|
|
}
|
|
key := libcommon.Hash{123}
|
|
emptyVal := uint256.NewInt(0)
|
|
|
|
val := uint256.NewInt(0).SetBytes([]byte("block 1"))
|
|
writeStorageBlockData(t, NewPlainStateWriter(tx, tx, 1), []storageData{
|
|
{
|
|
addr: addrs[0],
|
|
inc: 1,
|
|
key: key,
|
|
oldVal: emptyVal,
|
|
newVal: val,
|
|
},
|
|
{
|
|
addr: addrs[1],
|
|
inc: 1,
|
|
key: key,
|
|
oldVal: emptyVal,
|
|
newVal: val,
|
|
},
|
|
{
|
|
addr: addrs[2],
|
|
inc: 1,
|
|
key: key,
|
|
oldVal: emptyVal,
|
|
newVal: val,
|
|
},
|
|
})
|
|
|
|
prev := val
|
|
for i := 2; i < 1100; i++ {
|
|
val = uint256.NewInt(0).SetBytes([]byte("block " + strconv.Itoa(i)))
|
|
writeStorageBlockData(t, NewPlainStateWriter(tx, tx, uint64(i)), []storageData{
|
|
{
|
|
addr: addrs[0],
|
|
inc: 1,
|
|
key: key,
|
|
oldVal: prev,
|
|
newVal: val,
|
|
},
|
|
{
|
|
addr: addrs[1],
|
|
inc: 1,
|
|
key: key,
|
|
oldVal: prev,
|
|
newVal: val,
|
|
},
|
|
{
|
|
addr: addrs[2],
|
|
inc: 1,
|
|
key: key,
|
|
oldVal: prev,
|
|
newVal: val,
|
|
},
|
|
})
|
|
prev = val
|
|
}
|
|
|
|
val = uint256.NewInt(0).SetBytes([]byte("block 1100"))
|
|
|
|
writeStorageBlockData(t, NewPlainStateWriter(tx, tx, 1100), []storageData{
|
|
{
|
|
addr: addrs[0],
|
|
inc: 1,
|
|
key: key,
|
|
oldVal: prev,
|
|
newVal: val,
|
|
},
|
|
{
|
|
addr: addrs[1],
|
|
inc: 1,
|
|
key: key,
|
|
oldVal: prev,
|
|
newVal: val,
|
|
},
|
|
{
|
|
addr: addrs[2],
|
|
inc: 1,
|
|
key: key,
|
|
oldVal: prev,
|
|
newVal: val,
|
|
},
|
|
})
|
|
|
|
for _, blockNum := range []uint64{5, 100, 1000, 1050} {
|
|
obtained := &historyv2.ChangeSet{
|
|
Changes: make([]historyv2.Change, 0),
|
|
}
|
|
|
|
for _, addr := range addrs {
|
|
if err := WalkAsOfStorage(tx, addr, historyv2read.DefaultIncarnation, libcommon.Hash{}, blockNum, func(kAddr, kLoc []byte, v []byte) (b bool, e error) {
|
|
if innerErr := obtained.Add(append(libcommon.CopyBytes(kAddr), kLoc...), libcommon.CopyBytes(v)); innerErr != nil {
|
|
t.Fatal(innerErr)
|
|
}
|
|
return true, nil
|
|
}); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
}
|
|
valBytes := uint256.NewInt(0).SetBytes([]byte("block " + strconv.FormatUint(blockNum-1, 10))).Bytes()
|
|
expected := &historyv2.ChangeSet{
|
|
Changes: []historyv2.Change{
|
|
{
|
|
Key: append(addrs[0].Bytes(), key.Bytes()...),
|
|
Value: valBytes,
|
|
},
|
|
{
|
|
Key: append(addrs[1].Bytes(), key.Bytes()...),
|
|
Value: valBytes,
|
|
},
|
|
{
|
|
Key: append(addrs[2].Bytes(), key.Bytes()...),
|
|
Value: valBytes,
|
|
},
|
|
},
|
|
}
|
|
assertChangesEquals(t, obtained, expected)
|
|
}
|
|
}
|
|
|
|
type accData struct {
|
|
addr libcommon.Address
|
|
oldVal *accounts.Account
|
|
newVal *accounts.Account
|
|
}
|
|
|
|
func writeBlockData(t *testing.T, blockWriter *PlainStateWriter, data []accData) {
|
|
for i := range data {
|
|
if data[i].newVal != nil {
|
|
if err := blockWriter.UpdateAccountData(data[i].addr, data[i].oldVal, data[i].newVal); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
} else {
|
|
if err := blockWriter.DeleteAccount(data[i].addr, data[i].oldVal); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
}
|
|
}
|
|
|
|
if err := blockWriter.WriteChangeSets(); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if err := blockWriter.WriteHistory(); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
}
|
|
|
|
type storageData struct {
|
|
addr libcommon.Address
|
|
inc uint64
|
|
key libcommon.Hash
|
|
oldVal *uint256.Int
|
|
newVal *uint256.Int
|
|
}
|
|
|
|
func writeStorageBlockData(t *testing.T, blockWriter *PlainStateWriter, data []storageData) {
|
|
|
|
for i := range data {
|
|
if err := blockWriter.WriteAccountStorage(data[i].addr, data[i].inc, &data[i].key, data[i].oldVal, data[i].newVal); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
}
|
|
|
|
if err := blockWriter.WriteChangeSets(); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if err := blockWriter.WriteHistory(); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
}
|
|
func assertChangesEquals(t *testing.T, changesObtained, changesExpected *historyv2.ChangeSet) {
|
|
t.Helper()
|
|
sort.Sort(changesObtained)
|
|
sort.Sort(changesExpected)
|
|
if !reflect.DeepEqual(changesObtained, changesExpected) {
|
|
fmt.Printf("expected: %+v", changesExpected)
|
|
fmt.Printf("obtained: %+v", changesObtained)
|
|
t.Fatal("block result is incorrect")
|
|
}
|
|
}
|