erigon-pulse/turbo/snapshotsync/freezeblocks/block_snapshots_test.go

236 lines
7.3 KiB
Go
Raw Normal View History

package freezeblocks
2021-11-16 08:33:41 +00:00
import (
"context"
"path/filepath"
2021-11-16 08:33:41 +00:00
"testing"
"testing/fstest"
2021-11-16 08:33:41 +00:00
"github.com/ledgerwatch/erigon-lib/compress"
2022-11-20 03:41:30 +00:00
"github.com/ledgerwatch/erigon-lib/downloader/snaptype"
2021-11-16 08:33:41 +00:00
"github.com/ledgerwatch/erigon-lib/recsplit"
"github.com/ledgerwatch/log/v3"
"github.com/stretchr/testify/require"
2021-12-21 14:12:32 +00:00
"github.com/ledgerwatch/erigon/common/math"
"github.com/ledgerwatch/erigon/eth/ethconfig"
"github.com/ledgerwatch/erigon/params"
"github.com/ledgerwatch/erigon/params/networkname"
"github.com/ledgerwatch/erigon/turbo/snapshotsync/snapcfg"
2021-11-16 08:33:41 +00:00
)
func createTestSegmentFile(t *testing.T, from, to uint64, name snaptype.Type, dir string, logger log.Logger) {
c, err := compress.NewCompressor(context.Background(), "test", filepath.Join(dir, snaptype.SegmentFileName(from, to, name)), dir, 100, 1, log.LvlDebug, logger)
2022-02-22 01:55:24 +00:00
require.NoError(t, err)
defer c.Close()
err = c.AddWord([]byte{1})
require.NoError(t, err)
err = c.Compress()
require.NoError(t, err)
idx, err := recsplit.NewRecSplit(recsplit.RecSplitArgs{
KeyCount: 1,
BucketSize: 10,
TmpDir: dir,
IndexFile: filepath.Join(dir, snaptype.IdxFileName(from, to, name.String())),
2022-02-22 01:55:24 +00:00
LeafSize: 8,
}, logger)
2022-02-22 01:55:24 +00:00
require.NoError(t, err)
2022-02-22 07:44:57 +00:00
defer idx.Close()
2022-02-22 01:55:24 +00:00
err = idx.AddKey([]byte{1}, 0)
require.NoError(t, err)
err = idx.Build()
require.NoError(t, err)
if name == snaptype.Transactions {
2021-11-16 08:33:41 +00:00
idx, err := recsplit.NewRecSplit(recsplit.RecSplitArgs{
KeyCount: 1,
BucketSize: 10,
TmpDir: dir,
IndexFile: filepath.Join(dir, snaptype.IdxFileName(from, to, snaptype.Transactions2Block.String())),
2021-11-16 08:33:41 +00:00
LeafSize: 8,
}, logger)
2022-02-22 01:55:24 +00:00
require.NoError(t, err)
2021-11-16 08:33:41 +00:00
err = idx.AddKey([]byte{1}, 0)
2022-02-22 01:55:24 +00:00
require.NoError(t, err)
2021-11-16 08:33:41 +00:00
err = idx.Build()
2022-02-22 01:55:24 +00:00
require.NoError(t, err)
2022-02-22 07:44:57 +00:00
defer idx.Close()
2022-02-22 01:55:24 +00:00
}
}
2022-03-12 08:27:55 +00:00
func TestMergeSnapshots(t *testing.T) {
logger := log.New()
2022-02-22 01:55:24 +00:00
dir, require := t.TempDir(), require.New(t)
createFile := func(from, to uint64) {
for _, snT := range snaptype.AllSnapshotTypes {
createTestSegmentFile(t, from, to, snT, dir, logger)
2022-02-22 01:55:24 +00:00
}
}
2022-02-25 04:42:54 +00:00
N := uint64(7)
2022-02-22 01:55:24 +00:00
createFile(0, 500_000)
2022-02-25 04:42:54 +00:00
for i := uint64(500_000); i < 500_000+N*100_000; i += 100_000 {
createFile(i, i+100_000)
2021-11-16 08:33:41 +00:00
}
s := NewRoSnapshots(ethconfig.BlocksFreezing{Enabled: true}, dir, logger)
2022-02-22 01:55:24 +00:00
defer s.Close()
require.NoError(s.ReopenFolder())
{
merger := NewMerger(dir, 1, log.LvlInfo, nil, params.MainnetChainConfig, nil, logger)
ranges := merger.FindMergeRanges(s.Ranges())
require.True(len(ranges) > 0)
err := merger.Merge(context.Background(), s, ranges, s.Dir(), false)
require.NoError(err)
}
expectedFileName := snaptype.SegmentFileName(500_000, 1_000_000, snaptype.Transactions)
2022-02-22 01:55:24 +00:00
d, err := compress.NewDecompressor(filepath.Join(dir, expectedFileName))
require.NoError(err)
defer d.Close()
a := d.Count()
2022-02-25 04:42:54 +00:00
require.Equal(5, a)
2022-02-22 01:55:24 +00:00
{
merger := NewMerger(dir, 1, log.LvlInfo, nil, params.MainnetChainConfig, nil, logger)
ranges := merger.FindMergeRanges(s.Ranges())
require.True(len(ranges) == 0)
err := merger.Merge(context.Background(), s, ranges, s.Dir(), false)
require.NoError(err)
}
2022-02-22 01:55:24 +00:00
expectedFileName = snaptype.SegmentFileName(1_100_000, 1_200_000, snaptype.Transactions)
2022-02-22 01:55:24 +00:00
d, err = compress.NewDecompressor(filepath.Join(dir, expectedFileName))
require.NoError(err)
defer d.Close()
a = d.Count()
require.Equal(1, a)
2022-02-22 01:55:24 +00:00
}
func TestCanRetire(t *testing.T) {
require := require.New(t)
cases := []struct {
inFrom, inTo, outFrom, outTo uint64
can bool
}{
{0, 1234, 0, 1000, true},
{1_000_000, 1_120_000, 1_000_000, 1_100_000, true},
{2_500_000, 4_100_000, 2_500_000, 3_000_000, true},
{2_500_000, 2_500_100, 2_500_000, 2_500_000, false},
{1_001_000, 2_000_000, 1_001_000, 1_002_000, true},
}
for _, tc := range cases {
from, to, can := canRetire(tc.inFrom, tc.inTo)
require.Equal(int(tc.outFrom), int(from))
require.Equal(int(tc.outTo), int(to))
require.Equal(tc.can, can, tc.inFrom, tc.inTo)
}
}
2022-02-22 01:55:24 +00:00
func TestOpenAllSnapshot(t *testing.T) {
logger := log.New()
2022-02-22 01:55:24 +00:00
dir, require := t.TempDir(), require.New(t)
chainSnapshotCfg := snapcfg.KnownCfg(networkname.MainnetChainName, nil, nil)
2022-02-22 01:55:24 +00:00
chainSnapshotCfg.ExpectBlocks = math.MaxUint64
cfg := ethconfig.BlocksFreezing{Enabled: true}
createFile := func(from, to uint64, name snaptype.Type) { createTestSegmentFile(t, from, to, name, dir, logger) }
s := NewRoSnapshots(cfg, dir, logger)
2021-12-21 14:12:32 +00:00
defer s.Close()
err := s.ReopenFolder()
2021-11-16 08:33:41 +00:00
require.NoError(err)
require.Equal(0, len(s.Headers.segments))
2021-11-16 08:33:41 +00:00
s.Close()
createFile(500_000, 1_000_000, snaptype.Bodies)
s = NewRoSnapshots(cfg, dir, logger)
2021-12-21 14:12:32 +00:00
defer s.Close()
require.Equal(0, len(s.Bodies.segments)) //because, no headers and transactions snapshot files are created
2021-11-16 08:33:41 +00:00
s.Close()
createFile(500_000, 1_000_000, snaptype.Headers)
createFile(500_000, 1_000_000, snaptype.Transactions)
s = NewRoSnapshots(cfg, dir, logger)
err = s.ReopenFolder()
require.NoError(err)
require.Equal(0, len(s.Headers.segments))
2021-11-16 08:33:41 +00:00
s.Close()
createFile(0, 500_000, snaptype.Bodies)
createFile(0, 500_000, snaptype.Headers)
createFile(0, 500_000, snaptype.Transactions)
s = NewRoSnapshots(cfg, dir, logger)
defer s.Close()
err = s.ReopenFolder()
require.NoError(err)
require.Equal(2, len(s.Headers.segments))
2021-11-16 08:33:41 +00:00
view := s.View()
defer view.Close()
seg, ok := view.TxsSegment(10)
2021-11-16 08:33:41 +00:00
require.True(ok)
require.Equal(int(seg.ranges.to), 500_000)
2021-11-16 08:33:41 +00:00
seg, ok = view.TxsSegment(500_000)
2021-11-16 08:33:41 +00:00
require.True(ok)
require.Equal(int(seg.ranges.to), 1_000_000)
2021-11-16 08:33:41 +00:00
_, ok = view.TxsSegment(1_000_000)
2021-11-16 08:33:41 +00:00
require.False(ok)
2021-12-21 14:12:32 +00:00
// Erigon may create new snapshots by itself - with high bigger than hardcoded ExpectedBlocks
// ExpectedBlocks - says only how much block must come from Torrent
chainSnapshotCfg.ExpectBlocks = 500_000 - 1
s = NewRoSnapshots(cfg, dir, logger)
err = s.ReopenFolder()
2021-12-21 14:12:32 +00:00
require.NoError(err)
defer s.Close()
require.Equal(2, len(s.Headers.segments))
2021-12-21 14:12:32 +00:00
createFile(500_000, 900_000, snaptype.Headers)
createFile(500_000, 900_000, snaptype.Bodies)
createFile(500_000, 900_000, snaptype.Transactions)
chainSnapshotCfg.ExpectBlocks = math.MaxUint64
s = NewRoSnapshots(cfg, dir, logger)
2021-12-21 14:12:32 +00:00
defer s.Close()
err = s.ReopenFolder()
2022-02-22 01:55:24 +00:00
require.NoError(err)
2021-11-16 08:33:41 +00:00
}
func TestParseCompressedFileName(t *testing.T) {
require := require.New(t)
fs := fstest.MapFS{
"a": &fstest.MapFile{},
"1-a": &fstest.MapFile{},
"1-2-a": &fstest.MapFile{},
"1-2-bodies.info": &fstest.MapFile{},
"1-2-bodies.seg": &fstest.MapFile{},
"v2-1-2-bodies.seg": &fstest.MapFile{},
"v0-1-2-bodies.seg": &fstest.MapFile{},
"v1-1-2-bodies.seg": &fstest.MapFile{},
}
stat := func(name string) string {
s, err := fs.Stat(name)
require.NoError(err)
return s.Name()
}
_, err := snaptype.ParseFileName("", stat("a"))
2021-11-16 08:33:41 +00:00
require.Error(err)
_, err = snaptype.ParseFileName("", stat("1-a"))
2021-11-16 08:33:41 +00:00
require.Error(err)
_, err = snaptype.ParseFileName("", stat("1-2-a"))
2021-11-16 08:33:41 +00:00
require.Error(err)
_, err = snaptype.ParseFileName("", stat("1-2-bodies.info"))
2021-11-16 08:33:41 +00:00
require.Error(err)
_, err = snaptype.ParseFileName("", stat("1-2-bodies.seg"))
2021-11-16 08:33:41 +00:00
require.Error(err)
_, err = snaptype.ParseFileName("", stat("v2-1-2-bodies.seg"))
2021-11-16 08:33:41 +00:00
require.Error(err)
_, err = snaptype.ParseFileName("", stat("v0-1-2-bodies.seg"))
2021-11-16 08:33:41 +00:00
require.Error(err)
f, err := snaptype.ParseFileName("", stat("v1-1-2-bodies.seg"))
2021-11-16 08:33:41 +00:00
require.NoError(err)
require.Equal(f.T, snaptype.Bodies)
require.Equal(1_000, int(f.From))
require.Equal(2_000, int(f.To))
2021-11-16 08:33:41 +00:00
}