mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-05 10:32:19 +00:00
436656a40f
it's step towards saving canonical and non-canonical bodies in same table (and txs also in same own table). to reduce write amplification (cheaper re-orgs) PR change: reading BaseTxNum from existing snapshots instead of DB DB will store in field body.BaseTxNum - non-canonical TxnID Snapshots will store only canonical TxNum in field body.BaseTxNum
187 lines
5.7 KiB
Go
187 lines
5.7 KiB
Go
package snapshotsync_test
|
|
|
|
import (
|
|
"math/big"
|
|
"testing"
|
|
|
|
"github.com/holiman/uint256"
|
|
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/params"
|
|
"github.com/ledgerwatch/erigon/rlp"
|
|
"github.com/ledgerwatch/erigon/turbo/snapshotsync"
|
|
"github.com/ledgerwatch/erigon/turbo/stages"
|
|
"github.com/ledgerwatch/log/v3"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
func TestDump(t *testing.T) {
|
|
m := createDumpTestKV(t, 5)
|
|
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 := snapshotsync.DumpTxs(m.Ctx, m.DB, 0, 10, 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 */, nil); err != nil {
|
|
return err
|
|
}
|
|
nonceList = append(nonceList, slot.Nonce)
|
|
}
|
|
return nil
|
|
})
|
|
require.NoError(err)
|
|
require.Equal(2*(5+1), systemTxs)
|
|
require.Equal([]uint64{0, 1, 2, 3, 4}, nonceList)
|
|
require.Equal(2*(5+1)+5, 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 := snapshotsync.DumpTxs(m.Ctx, m.DB, 2, 5, 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 */, nil); err != nil {
|
|
return err
|
|
}
|
|
nonceList = append(nonceList, slot.Nonce)
|
|
}
|
|
return nil
|
|
})
|
|
require.NoError(err)
|
|
require.Equal(2*3, systemTxs)
|
|
require.Equal([]uint64{1, 2, 3}, nonceList)
|
|
require.Equal(2*3+3, cnt)
|
|
})
|
|
t.Run("headers", func(t *testing.T) {
|
|
require := require.New(t)
|
|
var nonceList []uint64
|
|
err := snapshotsync.DumpHeaders(m.Ctx, m.DB, 0, 10, 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([]uint64{0, 1, 2, 3, 4, 5}, nonceList)
|
|
})
|
|
t.Run("headers_not_from_zero", func(t *testing.T) {
|
|
require := require.New(t)
|
|
var nonceList []uint64
|
|
err := snapshotsync.DumpHeaders(m.Ctx, m.DB, 2, 5, 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([]uint64{2, 3, 4}, nonceList)
|
|
})
|
|
t.Run("body", func(t *testing.T) {
|
|
require := require.New(t)
|
|
i := 0
|
|
txsAmount := uint64(0)
|
|
var baseIdList []uint64
|
|
firstTxNum := uint64(0)
|
|
err := snapshotsync.DumpBodies(m.Ctx, m.DB, 0, 2, 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(2, i)
|
|
require.Equal(5, int(txsAmount))
|
|
require.Equal([]uint64{0, 2}, baseIdList)
|
|
|
|
firstTxNum += txsAmount
|
|
i = 0
|
|
baseIdList = baseIdList[:0]
|
|
err = snapshotsync.DumpBodies(m.Ctx, m.DB, 2, 10, 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(4, i)
|
|
require.Equal(17, int(txsAmount))
|
|
require.Equal([]uint64{5, 8, 11, 14}, baseIdList)
|
|
})
|
|
t.Run("body_not_from_zero", func(t *testing.T) {
|
|
require := require.New(t)
|
|
i := 0
|
|
var baseIdList []uint64
|
|
firstTxNum := uint64(1000)
|
|
err := snapshotsync.DumpBodies(m.Ctx, m.DB, 2, 5, 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(3, i)
|
|
require.Equal([]uint64{1000, 1000 + 3, 1000 + 6}, baseIdList)
|
|
})
|
|
}
|
|
|
|
func createDumpTestKV(t *testing.T, chainSize int) *stages.MockSentry {
|
|
var (
|
|
key, _ = crypto.HexToECDSA("b71c71a67e1177ad4e901695e1b4b9ee17ae16c6668d313eac2f96dbcda3f291")
|
|
addr = crypto.PubkeyToAddress(key.PublicKey)
|
|
gspec = &types.Genesis{
|
|
Config: params.TestChainConfig,
|
|
Alloc: types.GenesisAlloc{addr: {Balance: big.NewInt(math.MaxInt64)}},
|
|
}
|
|
signer = types.LatestSigner(gspec.Config)
|
|
)
|
|
m := stages.MockWithGenesis(t, gspec, key, 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)
|
|
}, false)
|
|
if err != nil {
|
|
t.Error(err)
|
|
}
|
|
// Construct testing chain
|
|
if err = m.InsertChain(chain); err != nil {
|
|
t.Error(err)
|
|
}
|
|
|
|
return m
|
|
}
|