erigon-pulse/trie/resolver_stateless_test.go

133 lines
3.2 KiB
Go

package trie
import (
"bytes"
"fmt"
"testing"
"github.com/ledgerwatch/turbo-geth/common"
)
type testWitnessStorage []byte
func (s *testWitnessStorage) GetWitnessesForBlock(_ uint64, _ uint32) ([]byte, error) {
return []byte(*s), nil
}
func generateKey(i int) []byte {
return []byte(fmt.Sprintf("key-number-%05d", i))
}
func generateValue(i int) []byte {
return []byte(fmt.Sprintf("value-number-%05d", i))
}
func buildTestTrie(numberOfNodes int) *Trie {
trie := New(EmptyRoot)
for i := 0; i < numberOfNodes; i++ {
trie.Update(generateKey(i), generateValue(i), 1)
}
return trie
}
func TestRebuildTrie(t *testing.T) {
trie1 := buildTestTrie(0)
trie2 := buildTestTrie(10)
trie3 := buildTestTrie(100)
w1, err := extractWitnessFromRootNode(trie1.root, 1, false, nil, nil)
if err != nil {
t.Error(err)
}
w2, err := extractWitnessFromRootNode(trie2.root, 1, false, nil, nil)
if err != nil {
t.Error(err)
}
w3, err := extractWitnessFromRootNode(trie3.root, 1, false, nil, nil)
if err != nil {
t.Error(err)
}
var buff bytes.Buffer
_, err = w1.WriteTo(&buff)
if err != nil {
t.Error(err)
}
err = buff.WriteByte(byte(OpNewTrie))
if err != nil {
t.Error(err)
}
_, err = w2.WriteTo(&buff)
if err != nil {
t.Error(err)
}
err = buff.WriteByte(byte(OpNewTrie))
if err != nil {
t.Error(err)
}
_, err = w3.WriteTo(&buff)
if err != nil {
t.Error(err)
}
storage := testWitnessStorage(buff.Bytes())
resolvedTries := make([]*Trie, 3)
currentTrie := 0
req1 := trie1.NewResolveRequest(nil, []byte{0x01}, 1, trie1.Hash().Bytes())
req2 := trie2.NewResolveRequest(nil, []byte{0x02}, 1, trie2.Hash().Bytes())
req21 := trie2.NewResolveRequest(nil, []byte{0x02}, 1, trie2.Hash().Bytes())
req3 := trie3.NewResolveRequest(nil, []byte{0x03}, 1, trie3.Hash().Bytes())
req31 := trie3.NewResolveRequest(nil, []byte{0x03}, 1, trie3.Hash().Bytes())
hookFunction := func(req *ResolveRequest, root node, rootHash common.Hash) error {
trie := New(rootHash)
trie.root = root
resolvedTries[currentTrie] = trie
currentTrie++
if !bytes.Equal(req.resolveHash.hash(), rootHash.Bytes()) {
return fmt.Errorf("root hash mismatch: expected %x got %x",
req.resolveHash.hash(), rootHash.Bytes())
}
return nil
}
// it should ignore duplicate resolve requests
resolver := NewResolverStateless([]*ResolveRequest{req1, req2, req21}, hookFunction)
pos, err := resolver.RebuildTrie(&storage, 1, 1, 0)
if err != nil {
t.Error(err)
}
// we also support partial resolution with continuation (for storage tries)
// so basically we first resolve accounts, then storages separately
// but we still want to keep one entry in a DB per block, so we store the last read position
// and then use it as a start
resolver = NewResolverStateless([]*ResolveRequest{req3, req31}, hookFunction)
_, err = resolver.RebuildTrie(&storage, 1, 1, pos)
if err != nil {
t.Error(err)
}
if !bytes.Equal(resolvedTries[0].Hash().Bytes(), trie1.Hash().Bytes()) {
t.Errorf("tries are different")
}
if !bytes.Equal(resolvedTries[1].Hash().Bytes(), trie2.Hash().Bytes()) {
t.Errorf("tries are different")
}
if !bytes.Equal(resolvedTries[2].Hash().Bytes(), trie3.Hash().Bytes()) {
t.Errorf("tries are different")
}
}