mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-07 03:22:18 +00:00
280 lines
8.4 KiB
Go
280 lines
8.4 KiB
Go
package freezeblocks_test
|
|
|
|
import (
|
|
"math/big"
|
|
"testing"
|
|
|
|
"github.com/holiman/uint256"
|
|
"github.com/ledgerwatch/erigon-lib/chain/networkname"
|
|
"github.com/ledgerwatch/erigon-lib/chain/snapcfg"
|
|
"github.com/ledgerwatch/log/v3"
|
|
"github.com/stretchr/testify/require"
|
|
|
|
"github.com/ledgerwatch/erigon-lib/chain"
|
|
libcommon "github.com/ledgerwatch/erigon-lib/common"
|
|
types2 "github.com/ledgerwatch/erigon-lib/types"
|
|
|
|
"github.com/ledgerwatch/erigon/common/math"
|
|
"github.com/ledgerwatch/erigon/core"
|
|
"github.com/ledgerwatch/erigon/core/types"
|
|
"github.com/ledgerwatch/erigon/crypto"
|
|
"github.com/ledgerwatch/erigon/ethdb/prune"
|
|
"github.com/ledgerwatch/erigon/params"
|
|
"github.com/ledgerwatch/erigon/rlp"
|
|
"github.com/ledgerwatch/erigon/turbo/snapshotsync/freezeblocks"
|
|
"github.com/ledgerwatch/erigon/turbo/stages/mock"
|
|
)
|
|
|
|
func nonceRange(from, to int) []uint64 {
|
|
a := make([]uint64, to-from+1)
|
|
for i := range a {
|
|
a[i] = uint64(from + i)
|
|
}
|
|
return a
|
|
}
|
|
|
|
func baseIdRange(base, indexer, len int) []uint64 {
|
|
a := make([]uint64, len)
|
|
index := 0
|
|
|
|
for i := range a {
|
|
a[i] = uint64(base + index)
|
|
index += indexer
|
|
}
|
|
return a
|
|
}
|
|
|
|
func TestDump(t *testing.T) {
|
|
type test struct {
|
|
chainConfig *chain.Config
|
|
chainSize int
|
|
}
|
|
|
|
withConfig := func(config chain.Config, sprints map[string]uint64) *chain.Config {
|
|
bor := *config.Bor
|
|
config.Bor = &bor
|
|
config.Bor.Sprint = sprints
|
|
return &config
|
|
}
|
|
|
|
tests := []test{
|
|
{
|
|
chainSize: 5,
|
|
chainConfig: params.TestChainConfig,
|
|
},
|
|
{
|
|
chainSize: 50,
|
|
chainConfig: params.TestChainConfig,
|
|
},
|
|
{
|
|
chainSize: 1000,
|
|
chainConfig: params.BorDevnetChainConfig,
|
|
},
|
|
{
|
|
chainSize: 2000,
|
|
chainConfig: params.BorDevnetChainConfig,
|
|
},
|
|
{
|
|
chainSize: 1000,
|
|
chainConfig: withConfig(*params.BorDevnetChainConfig,
|
|
map[string]uint64{
|
|
"0": 64,
|
|
"800": 16,
|
|
"1600": 8,
|
|
}),
|
|
},
|
|
{
|
|
chainSize: 2000,
|
|
chainConfig: withConfig(*params.BorDevnetChainConfig,
|
|
map[string]uint64{
|
|
"0": 64,
|
|
"800": 16,
|
|
"1600": 8,
|
|
}),
|
|
},
|
|
}
|
|
|
|
for _, test := range tests {
|
|
m := createDumpTestKV(t, test.chainConfig, test.chainSize)
|
|
chainID, _ := uint256.FromBig(m.ChainConfig.ChainID)
|
|
t.Run("txs", func(t *testing.T) {
|
|
require := require.New(t)
|
|
slot := types2.TxSlot{}
|
|
parseCtx := types2.NewTxParseContext(*chainID)
|
|
parseCtx.WithSender(false)
|
|
var sender [20]byte
|
|
|
|
var systemTxs int
|
|
var nonceList []uint64
|
|
cnt, err := freezeblocks.DumpTxs(m.Ctx, m.DB, 0, uint64(2*test.chainSize), m.ChainConfig, 1, log.LvlInfo, log.New(), func(v []byte) error {
|
|
if v == nil {
|
|
systemTxs++
|
|
} else {
|
|
if _, err := parseCtx.ParseTransaction(v[1+20:], 0, &slot, sender[:], false /* hasEnvelope */, false /* wrappedWithBlobs */, nil); err != nil {
|
|
return err
|
|
}
|
|
nonceList = append(nonceList, slot.Nonce)
|
|
}
|
|
return nil
|
|
})
|
|
require.NoError(err)
|
|
require.Equal(2*(test.chainSize+1), systemTxs)
|
|
require.Equal(nonceRange(0, test.chainSize-1), nonceList)
|
|
require.Equal(2*(test.chainSize+1)+test.chainSize, cnt)
|
|
})
|
|
t.Run("txs_not_from_zero", func(t *testing.T) {
|
|
require := require.New(t)
|
|
slot := types2.TxSlot{}
|
|
parseCtx := types2.NewTxParseContext(*chainID)
|
|
parseCtx.WithSender(false)
|
|
var sender [20]byte
|
|
|
|
var systemTxs int
|
|
var nonceList []uint64
|
|
cnt, err := freezeblocks.DumpTxs(m.Ctx, m.DB, 2, uint64(test.chainSize), m.ChainConfig, 1, log.LvlInfo, log.New(), func(v []byte) error {
|
|
if v == nil {
|
|
systemTxs++
|
|
} else {
|
|
if _, err := parseCtx.ParseTransaction(v[1+20:], 0, &slot, sender[:], false /* hasEnvelope */, false /* wrappedWithBlobs */, nil); err != nil {
|
|
return err
|
|
}
|
|
nonceList = append(nonceList, slot.Nonce)
|
|
}
|
|
return nil
|
|
})
|
|
require.NoError(err)
|
|
require.Equal(2*(test.chainSize-2), systemTxs)
|
|
require.Equal(nonceRange(1, test.chainSize-2), nonceList)
|
|
require.Equal(3*test.chainSize-6, cnt)
|
|
})
|
|
t.Run("headers", func(t *testing.T) {
|
|
require := require.New(t)
|
|
var nonceList []uint64
|
|
err := freezeblocks.DumpHeaders(m.Ctx, m.DB, 0, uint64(2*test.chainSize), 1, log.LvlInfo, log.New(), func(v []byte) error {
|
|
h := types.Header{}
|
|
if err := rlp.DecodeBytes(v[1:], &h); err != nil {
|
|
return err
|
|
}
|
|
nonceList = append(nonceList, h.Number.Uint64())
|
|
return nil
|
|
})
|
|
require.NoError(err)
|
|
require.Equal(nonceRange(0, test.chainSize), nonceList)
|
|
})
|
|
t.Run("headers_not_from_zero", func(t *testing.T) {
|
|
require := require.New(t)
|
|
var nonceList []uint64
|
|
err := freezeblocks.DumpHeaders(m.Ctx, m.DB, 2, uint64(test.chainSize), 1, log.LvlInfo, log.New(), func(v []byte) error {
|
|
h := types.Header{}
|
|
if err := rlp.DecodeBytes(v[1:], &h); err != nil {
|
|
return err
|
|
}
|
|
nonceList = append(nonceList, h.Number.Uint64())
|
|
return nil
|
|
})
|
|
require.NoError(err)
|
|
require.Equal(nonceRange(2, test.chainSize-1), nonceList)
|
|
})
|
|
t.Run("body", func(t *testing.T) {
|
|
require := require.New(t)
|
|
i := 0
|
|
txsAmount := uint64(0)
|
|
var baseIdList []uint64
|
|
firstTxNum := uint64(0)
|
|
err := freezeblocks.DumpBodies(m.Ctx, m.DB, 0, uint64(test.chainSize-3), firstTxNum, 1, log.LvlInfo, log.New(), func(v []byte) error {
|
|
i++
|
|
body := &types.BodyForStorage{}
|
|
require.NoError(rlp.DecodeBytes(v, body))
|
|
txsAmount += uint64(body.TxAmount)
|
|
baseIdList = append(baseIdList, body.BaseTxId)
|
|
return nil
|
|
})
|
|
require.NoError(err)
|
|
require.Equal(test.chainSize-3, i)
|
|
require.Equal(3*(test.chainSize-3)-1, int(txsAmount))
|
|
require.Equal(append([]uint64{0}, baseIdRange(2, 3, test.chainSize-4)...), baseIdList)
|
|
|
|
firstTxNum += txsAmount
|
|
i = 0
|
|
baseIdList = baseIdList[:0]
|
|
err = freezeblocks.DumpBodies(m.Ctx, m.DB, 2, uint64(2*test.chainSize), firstTxNum, 1, log.LvlInfo, log.New(), func(v []byte) error {
|
|
i++
|
|
body := &types.BodyForStorage{}
|
|
require.NoError(rlp.DecodeBytes(v, body))
|
|
txsAmount += uint64(body.TxAmount)
|
|
baseIdList = append(baseIdList, body.BaseTxId)
|
|
return nil
|
|
})
|
|
require.NoError(err)
|
|
require.Equal(test.chainSize-1, i)
|
|
require.Equal(firstTxNum+uint64(3*(test.chainSize-1)), txsAmount)
|
|
require.Equal(baseIdRange(int(firstTxNum), 3, test.chainSize-1), baseIdList)
|
|
})
|
|
t.Run("body_not_from_zero", func(t *testing.T) {
|
|
require := require.New(t)
|
|
i := 0
|
|
var baseIdList []uint64
|
|
firstTxNum := uint64(1000)
|
|
err := freezeblocks.DumpBodies(m.Ctx, m.DB, 2, uint64(test.chainSize), firstTxNum, 1, log.LvlInfo, log.New(), func(v []byte) error {
|
|
i++
|
|
body := &types.BodyForStorage{}
|
|
require.NoError(rlp.DecodeBytes(v, body))
|
|
baseIdList = append(baseIdList, body.BaseTxId)
|
|
return nil
|
|
})
|
|
require.NoError(err)
|
|
require.Equal(test.chainSize-2, i)
|
|
require.Equal(baseIdRange(int(firstTxNum), 3, test.chainSize-2), baseIdList)
|
|
})
|
|
t.Run("blocks", func(t *testing.T) {
|
|
if test.chainSize < 1000 || test.chainSize%1000 != 0 {
|
|
t.Skip("Block dump requires chain size to be a multiple of 1000")
|
|
}
|
|
|
|
require := require.New(t)
|
|
|
|
logger := log.New()
|
|
|
|
tmpDir, snapDir := t.TempDir(), t.TempDir()
|
|
snConfig := snapcfg.KnownCfg(networkname.MainnetChainName, nil, nil)
|
|
snConfig.ExpectBlocks = math.MaxUint64
|
|
|
|
err := freezeblocks.DumpBlocks(m.Ctx, 0, uint64(test.chainSize), uint64(test.chainSize), tmpDir, snapDir, 0, m.DB, 1, log.LvlInfo, logger, m.BlockReader)
|
|
require.NoError(err)
|
|
})
|
|
}
|
|
}
|
|
|
|
func createDumpTestKV(t *testing.T, chainConfig *chain.Config, chainSize int) *mock.MockSentry {
|
|
var (
|
|
key, _ = crypto.HexToECDSA("b71c71a67e1177ad4e901695e1b4b9ee17ae16c6668d313eac2f96dbcda3f291")
|
|
addr = crypto.PubkeyToAddress(key.PublicKey)
|
|
gspec = &types.Genesis{
|
|
Config: chainConfig,
|
|
Alloc: types.GenesisAlloc{addr: {Balance: (&big.Int{}).Mul(big.NewInt(math.MaxInt64), big.NewInt(int64(chainSize)))}},
|
|
}
|
|
signer = types.LatestSigner(gspec.Config)
|
|
)
|
|
|
|
m := mock.MockWithGenesisPruneMode(t, gspec, key, chainSize, prune.DefaultMode, false)
|
|
|
|
// Generate testing blocks
|
|
chain, err := core.GenerateChain(m.ChainConfig, m.Genesis, m.Engine, m.DB, chainSize, func(i int, b *core.BlockGen) {
|
|
b.SetCoinbase(libcommon.Address{1})
|
|
tx, txErr := types.SignTx(types.NewTransaction(b.TxNonce(addr), libcommon.HexToAddress("deadbeef"), uint256.NewInt(100), 21000, uint256.NewInt(uint64(int64(i+1)*params.GWei)), nil), *signer, key)
|
|
if txErr != nil {
|
|
t.Fatalf("failed to create tx: %v", txErr)
|
|
}
|
|
b.AddTx(tx)
|
|
})
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
// Construct testing chain
|
|
if err = m.InsertChain(chain); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
return m
|
|
}
|