2021-11-13 12:12:29 +00:00
|
|
|
/*
|
2022-01-16 20:27:44 +00:00
|
|
|
Copyright 2022 Erigon contributors
|
2021-11-13 12:12:29 +00:00
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
package aggregator
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"encoding/binary"
|
|
|
|
"testing"
|
|
|
|
|
2021-11-16 00:11:40 +00:00
|
|
|
"github.com/holiman/uint256"
|
2021-11-13 12:12:29 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func int160(i uint64) []byte {
|
|
|
|
b := make([]byte, 20)
|
|
|
|
binary.BigEndian.PutUint64(b[12:], i)
|
|
|
|
return b
|
|
|
|
}
|
|
|
|
|
|
|
|
func int256(i uint64) []byte {
|
|
|
|
b := make([]byte, 32)
|
|
|
|
binary.BigEndian.PutUint64(b[24:], i)
|
|
|
|
return b
|
|
|
|
}
|
|
|
|
|
2022-01-12 11:21:52 +00:00
|
|
|
func accountWithBalance(i uint64) []byte {
|
|
|
|
balance := uint256.NewInt(i)
|
|
|
|
var l int
|
|
|
|
l++
|
|
|
|
l++
|
|
|
|
if i > 0 {
|
|
|
|
l += balance.ByteLen()
|
|
|
|
}
|
|
|
|
l++
|
|
|
|
l++
|
|
|
|
value := make([]byte, l)
|
|
|
|
pos := 0
|
|
|
|
value[pos] = 0
|
|
|
|
pos++
|
|
|
|
if balance.IsZero() {
|
|
|
|
value[pos] = 0
|
|
|
|
pos++
|
|
|
|
} else {
|
|
|
|
balanceBytes := balance.ByteLen()
|
|
|
|
value[pos] = byte(balanceBytes)
|
|
|
|
pos++
|
|
|
|
balance.WriteToSlice(value[pos : pos+balanceBytes])
|
|
|
|
pos += balanceBytes
|
|
|
|
}
|
|
|
|
value[pos] = 0
|
|
|
|
pos++
|
|
|
|
value[pos] = 0
|
|
|
|
return value
|
|
|
|
}
|
|
|
|
|
2021-11-13 12:12:29 +00:00
|
|
|
func TestSimpleAggregator(t *testing.T) {
|
|
|
|
tmpDir := t.TempDir()
|
2022-02-16 16:44:00 +00:00
|
|
|
a, err := NewAggregator(tmpDir, 16, 4, true, true)
|
2021-11-13 12:12:29 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2022-01-16 20:27:44 +00:00
|
|
|
w := a.MakeStateWriter(true /* beforeOn */)
|
2022-02-03 08:49:47 +00:00
|
|
|
if err = w.Reset(0); err != nil {
|
2021-11-13 12:12:29 +00:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2022-01-12 11:21:52 +00:00
|
|
|
defer w.Close()
|
|
|
|
var account1 = accountWithBalance(1)
|
2022-02-03 08:49:47 +00:00
|
|
|
w.UpdateAccountData(int160(1), account1, false /* trace */)
|
2022-01-12 11:21:52 +00:00
|
|
|
if err = w.FinishTx(0, false); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2022-01-16 20:27:44 +00:00
|
|
|
if err = w.Aggregate(false /* trace */); err != nil {
|
2021-11-13 12:12:29 +00:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2022-02-03 08:49:47 +00:00
|
|
|
r := a.MakeStateReader(2)
|
|
|
|
acc := r.ReadAccountData(int160(1), false /* trace */)
|
2021-11-13 12:12:29 +00:00
|
|
|
if !bytes.Equal(acc, account1) {
|
|
|
|
t.Errorf("read account %x, expected account %x", acc, account1)
|
|
|
|
}
|
|
|
|
a.Close()
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestLoopAggregator(t *testing.T) {
|
|
|
|
tmpDir := t.TempDir()
|
2022-02-16 16:44:00 +00:00
|
|
|
a, err := NewAggregator(tmpDir, 16, 4, true, true)
|
2021-11-13 12:12:29 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2021-11-16 00:11:40 +00:00
|
|
|
defer a.Close()
|
2022-01-12 11:21:52 +00:00
|
|
|
var account1 = accountWithBalance(1)
|
2022-01-16 20:27:44 +00:00
|
|
|
w := a.MakeStateWriter(true /* beforeOn */)
|
2022-01-12 11:21:52 +00:00
|
|
|
defer w.Close()
|
2021-11-13 12:12:29 +00:00
|
|
|
for blockNum := uint64(0); blockNum < 1000; blockNum++ {
|
|
|
|
accountKey := int160(blockNum/10 + 1)
|
|
|
|
//fmt.Printf("blockNum = %d\n", blockNum)
|
2022-02-03 08:49:47 +00:00
|
|
|
if err = w.Reset(blockNum); err != nil {
|
2021-11-13 12:12:29 +00:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2022-02-03 08:49:47 +00:00
|
|
|
w.UpdateAccountData(accountKey, account1, false /* trace */)
|
2022-01-12 11:21:52 +00:00
|
|
|
if err = w.FinishTx(blockNum, false /* trace */); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2022-01-16 20:27:44 +00:00
|
|
|
if err = w.Aggregate(false /* trace */); err != nil {
|
2021-11-13 12:12:29 +00:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2022-02-03 08:49:47 +00:00
|
|
|
r := a.MakeStateReader(blockNum + 1)
|
|
|
|
acc := r.ReadAccountData(accountKey, false /* trace */)
|
2021-11-13 12:12:29 +00:00
|
|
|
if !bytes.Equal(acc, account1) {
|
|
|
|
t.Errorf("read account %x, expected account %x for block %d", acc, account1, blockNum)
|
|
|
|
}
|
2022-01-12 11:21:52 +00:00
|
|
|
account1 = accountWithBalance(blockNum + 2)
|
2021-11-13 12:12:29 +00:00
|
|
|
}
|
|
|
|
blockNum := uint64(1000)
|
2022-02-03 08:49:47 +00:00
|
|
|
r := a.MakeStateReader(blockNum)
|
2021-11-13 12:12:29 +00:00
|
|
|
for i := uint64(0); i < blockNum/10+1; i++ {
|
|
|
|
accountKey := int160(i)
|
|
|
|
var expected []byte
|
|
|
|
if i > 0 {
|
2022-01-12 11:21:52 +00:00
|
|
|
expected = accountWithBalance(i * 10)
|
2021-11-13 12:12:29 +00:00
|
|
|
}
|
2022-02-03 08:49:47 +00:00
|
|
|
acc := r.ReadAccountData(accountKey, false /* trace */)
|
2021-11-13 12:12:29 +00:00
|
|
|
if !bytes.Equal(acc, expected) {
|
|
|
|
t.Errorf("read account %x, expected account %x for block %d", acc, expected, i)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-11-16 00:11:40 +00:00
|
|
|
|
|
|
|
func TestRecreateAccountWithStorage(t *testing.T) {
|
|
|
|
tmpDir := t.TempDir()
|
2022-02-16 16:44:00 +00:00
|
|
|
a, err := NewAggregator(tmpDir, 16, 4, true, true)
|
2021-11-16 00:11:40 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer a.Close()
|
|
|
|
accountKey := int160(1)
|
2022-01-12 11:21:52 +00:00
|
|
|
var account1 = accountWithBalance(1)
|
|
|
|
var account2 = accountWithBalance(2)
|
2022-01-16 20:27:44 +00:00
|
|
|
w := a.MakeStateWriter(true /* beforeOn */)
|
2022-01-12 11:21:52 +00:00
|
|
|
defer w.Close()
|
2021-11-16 00:11:40 +00:00
|
|
|
for blockNum := uint64(0); blockNum < 100; blockNum++ {
|
2022-02-03 08:49:47 +00:00
|
|
|
if err = w.Reset(blockNum); err != nil {
|
2021-11-16 00:11:40 +00:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
switch blockNum {
|
|
|
|
case 1:
|
2022-02-03 08:49:47 +00:00
|
|
|
w.UpdateAccountData(accountKey, account1, false /* trace */)
|
2021-11-16 00:11:40 +00:00
|
|
|
for s := uint64(0); s < 100; s++ {
|
2022-02-03 08:49:47 +00:00
|
|
|
w.WriteAccountStorage(accountKey, int256(s), uint256.NewInt(s+1), false /* trace */)
|
2021-11-16 00:11:40 +00:00
|
|
|
}
|
|
|
|
case 22:
|
2022-02-03 08:49:47 +00:00
|
|
|
w.DeleteAccount(accountKey, false /* trace */)
|
2021-11-16 21:59:26 +00:00
|
|
|
case 45:
|
2022-02-03 08:49:47 +00:00
|
|
|
w.UpdateAccountData(accountKey, account2, false /* trace */)
|
2021-11-16 21:59:26 +00:00
|
|
|
for s := uint64(50); s < 150; s++ {
|
2022-02-03 08:49:47 +00:00
|
|
|
w.WriteAccountStorage(accountKey, int256(s), uint256.NewInt(2*s+1), false /* trace */)
|
2021-11-16 21:59:26 +00:00
|
|
|
}
|
2021-11-16 00:11:40 +00:00
|
|
|
}
|
2022-01-12 11:21:52 +00:00
|
|
|
if err = w.FinishTx(blockNum, false /* trace */); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2022-01-16 20:27:44 +00:00
|
|
|
if err = w.Aggregate(false /* trace */); err != nil {
|
2021-11-16 00:11:40 +00:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2022-02-03 08:49:47 +00:00
|
|
|
r := a.MakeStateReader(blockNum + 1)
|
2021-11-16 00:11:40 +00:00
|
|
|
switch blockNum {
|
|
|
|
case 1:
|
2022-02-03 08:49:47 +00:00
|
|
|
acc := r.ReadAccountData(accountKey, false /* trace */)
|
2021-11-16 00:11:40 +00:00
|
|
|
if !bytes.Equal(account1, acc) {
|
|
|
|
t.Errorf("wrong account after block %d, expected %x, got %x", blockNum, account1, acc)
|
|
|
|
}
|
|
|
|
for s := uint64(0); s < 100; s++ {
|
2022-02-03 08:49:47 +00:00
|
|
|
v := r.ReadAccountStorage(accountKey, int256(s), false /* trace */)
|
2021-11-16 00:11:40 +00:00
|
|
|
if !uint256.NewInt(s + 1).Eq(v) {
|
2021-11-16 21:59:26 +00:00
|
|
|
t.Errorf("wrong storage value after block %d, expected %d, got %s", blockNum, s+1, v)
|
2021-11-16 00:11:40 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
case 22, 44:
|
2022-02-03 08:49:47 +00:00
|
|
|
acc := r.ReadAccountData(accountKey, false /* trace */)
|
2021-11-16 00:11:40 +00:00
|
|
|
if len(acc) > 0 {
|
|
|
|
t.Errorf("wrong account after block %d, expected nil, got %x", blockNum, acc)
|
|
|
|
}
|
|
|
|
for s := uint64(0); s < 100; s++ {
|
2022-02-03 08:49:47 +00:00
|
|
|
v := r.ReadAccountStorage(accountKey, int256(s), false /* trace */)
|
2021-11-16 00:11:40 +00:00
|
|
|
if v != nil {
|
|
|
|
t.Errorf("wrong storage value after block %d, expected nil, got %s", blockNum, v)
|
|
|
|
}
|
|
|
|
}
|
2021-11-16 21:59:26 +00:00
|
|
|
case 66:
|
2022-02-03 08:49:47 +00:00
|
|
|
acc := r.ReadAccountData(accountKey, false /* trace */)
|
2021-11-16 21:59:26 +00:00
|
|
|
if !bytes.Equal(account2, acc) {
|
|
|
|
t.Errorf("wrong account after block %d, expected %x, got %x", blockNum, account1, acc)
|
|
|
|
}
|
|
|
|
for s := uint64(0); s < 150; s++ {
|
2022-02-03 08:49:47 +00:00
|
|
|
v := r.ReadAccountStorage(accountKey, int256(s), false /* trace */)
|
2021-11-16 21:59:26 +00:00
|
|
|
if s < 50 {
|
|
|
|
if v != nil {
|
|
|
|
t.Errorf("wrong storage value after block %d, expected nil, got %s", blockNum, v)
|
|
|
|
}
|
|
|
|
} else if v == nil || !uint256.NewInt(2*s+1).Eq(v) {
|
|
|
|
t.Errorf("wrong storage value after block %d, expected %d, got %s", blockNum, 2*s+1, v)
|
|
|
|
}
|
|
|
|
}
|
2021-11-16 00:11:40 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-11-17 15:49:48 +00:00
|
|
|
|
|
|
|
func TestChangeCode(t *testing.T) {
|
|
|
|
tmpDir := t.TempDir()
|
2022-02-16 16:44:00 +00:00
|
|
|
a, err := NewAggregator(tmpDir, 16, 4, true, true)
|
2021-11-17 15:49:48 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer a.Close()
|
|
|
|
accountKey := int160(1)
|
2022-01-12 11:21:52 +00:00
|
|
|
var account1 = accountWithBalance(1)
|
2021-11-17 15:49:48 +00:00
|
|
|
var code1 = []byte("This is the code number 1")
|
2022-01-16 20:27:44 +00:00
|
|
|
w := a.MakeStateWriter(true /* beforeOn */)
|
2022-01-12 11:21:52 +00:00
|
|
|
defer w.Close()
|
2021-11-17 15:49:48 +00:00
|
|
|
for blockNum := uint64(0); blockNum < 100; blockNum++ {
|
2022-02-03 08:49:47 +00:00
|
|
|
if err = w.Reset(blockNum); err != nil {
|
2021-11-17 15:49:48 +00:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
switch blockNum {
|
|
|
|
case 1:
|
2022-02-03 08:49:47 +00:00
|
|
|
w.UpdateAccountData(accountKey, account1, false /* trace */)
|
|
|
|
w.UpdateAccountCode(accountKey, code1, false /* trace */)
|
2021-11-17 15:49:48 +00:00
|
|
|
case 25:
|
2022-02-03 08:49:47 +00:00
|
|
|
w.DeleteAccount(accountKey, false /* trace */)
|
2021-11-17 15:49:48 +00:00
|
|
|
}
|
2022-01-12 11:21:52 +00:00
|
|
|
if err = w.FinishTx(blockNum, false /* trace */); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2022-01-16 20:27:44 +00:00
|
|
|
if err = w.Aggregate(false /* trace */); err != nil {
|
2021-11-17 15:49:48 +00:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2022-02-03 08:49:47 +00:00
|
|
|
r := a.MakeStateReader(blockNum + 1)
|
2021-11-17 15:49:48 +00:00
|
|
|
switch blockNum {
|
|
|
|
case 22:
|
2022-02-03 08:49:47 +00:00
|
|
|
acc := r.ReadAccountData(accountKey, false /* trace */)
|
2021-11-17 15:49:48 +00:00
|
|
|
if !bytes.Equal(account1, acc) {
|
|
|
|
t.Errorf("wrong account after block %d, expected %x, got %x", blockNum, account1, acc)
|
|
|
|
}
|
2022-02-03 08:49:47 +00:00
|
|
|
code := r.ReadAccountCode(accountKey, false /* trace */)
|
2021-11-17 15:49:48 +00:00
|
|
|
if !bytes.Equal(code1, code) {
|
|
|
|
t.Errorf("wrong code after block %d, expected %x, got %x", blockNum, code1, code)
|
|
|
|
}
|
|
|
|
case 47:
|
2022-02-03 08:49:47 +00:00
|
|
|
code := r.ReadAccountCode(accountKey, false /* trace */)
|
2021-11-17 15:49:48 +00:00
|
|
|
if code != nil {
|
|
|
|
t.Errorf("wrong code after block %d, expected nil, got %x", blockNum, code)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|