2021-11-16 08:33:41 +00:00
|
|
|
package snapshotsync
|
|
|
|
|
|
|
|
import (
|
2022-01-18 08:01:36 +00:00
|
|
|
"context"
|
2022-02-23 03:54:35 +00:00
|
|
|
"os"
|
2022-02-12 13:33:09 +00:00
|
|
|
"path/filepath"
|
2021-11-16 08:33:41 +00:00
|
|
|
"testing"
|
2022-02-23 03:54:35 +00:00
|
|
|
"testing/fstest"
|
2021-11-16 08:33:41 +00:00
|
|
|
|
2022-02-22 09:02:09 +00:00
|
|
|
dir2 "github.com/ledgerwatch/erigon-lib/common/dir"
|
2021-11-16 08:33:41 +00:00
|
|
|
"github.com/ledgerwatch/erigon-lib/compress"
|
|
|
|
"github.com/ledgerwatch/erigon-lib/recsplit"
|
2021-12-21 14:12:32 +00:00
|
|
|
"github.com/ledgerwatch/erigon/common/math"
|
2022-01-20 05:01:02 +00:00
|
|
|
"github.com/ledgerwatch/erigon/eth/ethconfig"
|
2021-12-14 10:13:17 +00:00
|
|
|
"github.com/ledgerwatch/erigon/params/networkname"
|
|
|
|
"github.com/ledgerwatch/erigon/turbo/snapshotsync/snapshothashes"
|
2022-02-22 09:02:09 +00:00
|
|
|
"github.com/stretchr/testify/assert"
|
2021-11-16 08:33:41 +00:00
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
)
|
|
|
|
|
2022-02-23 03:54:35 +00:00
|
|
|
func createTestSegmentFile(t *testing.T, from, to uint64, name Type, dir string) {
|
2022-02-22 01:55:24 +00:00
|
|
|
c, err := compress.NewCompressor(context.Background(), "test", filepath.Join(dir, SegmentFileName(from, to, name)), dir, 100, 1)
|
|
|
|
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, IdxFileName(from, to, name)),
|
|
|
|
LeafSize: 8,
|
|
|
|
})
|
|
|
|
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 == Transactions {
|
2021-11-16 08:33:41 +00:00
|
|
|
idx, err := recsplit.NewRecSplit(recsplit.RecSplitArgs{
|
|
|
|
KeyCount: 1,
|
|
|
|
BucketSize: 10,
|
|
|
|
TmpDir: dir,
|
2022-02-22 01:55:24 +00:00
|
|
|
IndexFile: filepath.Join(dir, IdxFileName(from, to, Transactions2Block)),
|
2021-11-16 08:33:41 +00:00
|
|
|
LeafSize: 8,
|
|
|
|
})
|
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
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestMerge(t *testing.T) {
|
|
|
|
dir, require := t.TempDir(), require.New(t)
|
|
|
|
createFile := func(from, to uint64) {
|
|
|
|
for _, snT := range AllSnapshotTypes {
|
|
|
|
createTestSegmentFile(t, from, to, snT, dir)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
N := uint64(15)
|
|
|
|
createFile(0, 500_000)
|
|
|
|
for i := uint64(500_000); i < 500_000+N*50_000; i += 50_000 {
|
|
|
|
createFile(i, i+50_000)
|
2021-11-16 08:33:41 +00:00
|
|
|
}
|
2022-02-22 01:55:24 +00:00
|
|
|
cfg := ethconfig.Snapshot{Enabled: true}
|
|
|
|
s := NewRoSnapshots(cfg, dir)
|
|
|
|
defer s.Close()
|
|
|
|
|
|
|
|
require.NoError(s.ReopenSegments())
|
2022-02-24 07:15:03 +00:00
|
|
|
_, err := findAndMergeBlockSegments(context.Background(), s, dir, 1)
|
2022-02-22 01:55:24 +00:00
|
|
|
require.NoError(err)
|
|
|
|
require.NoError(s.ReopenSegments())
|
|
|
|
|
|
|
|
expectedFileName := SegmentFileName(500_000, 1_000_000, Transactions)
|
|
|
|
d, err := compress.NewDecompressor(filepath.Join(dir, expectedFileName))
|
|
|
|
require.NoError(err)
|
|
|
|
defer d.Close()
|
|
|
|
a := d.Count()
|
|
|
|
require.Equal(10, a)
|
|
|
|
|
2022-02-24 07:15:03 +00:00
|
|
|
_, err = findAndMergeBlockSegments(context.Background(), s, dir, 1)
|
2022-02-22 01:55:24 +00:00
|
|
|
require.NoError(err)
|
|
|
|
require.NoError(s.ReopenSegments())
|
|
|
|
|
|
|
|
expectedFileName = SegmentFileName(1_000_000, 1_250_000, Transactions)
|
|
|
|
d, err = compress.NewDecompressor(filepath.Join(dir, expectedFileName))
|
|
|
|
require.NoError(err)
|
|
|
|
defer d.Close()
|
|
|
|
a = d.Count()
|
|
|
|
require.Equal(5, a)
|
|
|
|
}
|
|
|
|
|
2022-02-22 09:02:09 +00:00
|
|
|
func TestRecompress(t *testing.T) {
|
|
|
|
dir, require := t.TempDir(), require.New(t)
|
|
|
|
createFile := func(from, to uint64) { createTestSegmentFile(t, from, to, Headers, dir) }
|
|
|
|
|
|
|
|
createFile(0, 1_000)
|
|
|
|
err := RecompressSegments(context.Background(), &dir2.Rw{Path: dir}, dir)
|
|
|
|
require.NoError(err)
|
|
|
|
|
|
|
|
d, err := compress.NewDecompressor(filepath.Join(dir, SegmentFileName(0, 1_000, Headers)))
|
|
|
|
require.NoError(err)
|
|
|
|
defer d.Close()
|
|
|
|
assert.Equal(t, 1, d.Count())
|
|
|
|
}
|
|
|
|
|
2022-02-22 01:55:24 +00:00
|
|
|
func TestOpenAllSnapshot(t *testing.T) {
|
|
|
|
dir, require := t.TempDir(), require.New(t)
|
|
|
|
chainSnapshotCfg := snapshothashes.KnownConfig(networkname.MainnetChainName)
|
|
|
|
chainSnapshotCfg.ExpectBlocks = math.MaxUint64
|
|
|
|
cfg := ethconfig.Snapshot{Enabled: true}
|
2022-02-23 03:54:35 +00:00
|
|
|
createFile := func(from, to uint64, name Type) { createTestSegmentFile(t, from, to, name, dir) }
|
2022-02-18 02:24:17 +00:00
|
|
|
s := NewRoSnapshots(cfg, dir)
|
2021-12-21 14:12:32 +00:00
|
|
|
defer s.Close()
|
2021-12-05 02:03:08 +00:00
|
|
|
err := s.ReopenSegments()
|
2021-11-16 08:33:41 +00:00
|
|
|
require.NoError(err)
|
|
|
|
require.Equal(0, len(s.blocks))
|
|
|
|
s.Close()
|
|
|
|
|
|
|
|
createFile(500_000, 1_000_000, Bodies)
|
2022-02-18 02:24:17 +00:00
|
|
|
s = NewRoSnapshots(cfg, dir)
|
2021-12-21 14:12:32 +00:00
|
|
|
defer s.Close()
|
2021-11-16 08:33:41 +00:00
|
|
|
require.Equal(0, len(s.blocks)) //because, no headers and transactions snapshot files are created
|
|
|
|
s.Close()
|
|
|
|
|
|
|
|
createFile(500_000, 1_000_000, Headers)
|
|
|
|
createFile(500_000, 1_000_000, Transactions)
|
2022-02-18 02:24:17 +00:00
|
|
|
s = NewRoSnapshots(cfg, dir)
|
2021-12-05 02:03:08 +00:00
|
|
|
err = s.ReopenSegments()
|
2021-12-21 14:12:32 +00:00
|
|
|
require.Error(err)
|
2021-11-16 08:33:41 +00:00
|
|
|
require.Equal(0, len(s.blocks)) //because, no gaps are allowed (expect snapshots from block 0)
|
|
|
|
s.Close()
|
|
|
|
|
|
|
|
createFile(0, 500_000, Bodies)
|
|
|
|
createFile(0, 500_000, Headers)
|
|
|
|
createFile(0, 500_000, Transactions)
|
2022-02-18 02:24:17 +00:00
|
|
|
s = NewRoSnapshots(cfg, dir)
|
2022-01-26 15:26:33 +00:00
|
|
|
defer s.Close()
|
|
|
|
|
2021-12-05 02:03:08 +00:00
|
|
|
err = s.ReopenSegments()
|
|
|
|
require.NoError(err)
|
2022-01-26 15:26:33 +00:00
|
|
|
s.indicesReady.Store(true)
|
2021-11-16 08:33:41 +00:00
|
|
|
require.Equal(2, len(s.blocks))
|
|
|
|
|
|
|
|
sn, ok := s.Blocks(10)
|
|
|
|
require.True(ok)
|
|
|
|
require.Equal(int(sn.To), 500_000)
|
|
|
|
|
|
|
|
sn, ok = s.Blocks(500_000)
|
|
|
|
require.True(ok)
|
|
|
|
require.Equal(int(sn.To), 1_000_000) // [from:to)
|
|
|
|
|
|
|
|
_, ok = s.Blocks(1_000_000)
|
|
|
|
require.False(ok)
|
2021-12-21 14:12:32 +00:00
|
|
|
|
2022-01-22 04:18:24 +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
|
2022-01-20 05:01:02 +00:00
|
|
|
chainSnapshotCfg.ExpectBlocks = 500_000 - 1
|
2022-02-18 02:24:17 +00:00
|
|
|
s = NewRoSnapshots(cfg, dir)
|
2021-12-21 14:12:32 +00:00
|
|
|
err = s.ReopenSegments()
|
|
|
|
require.NoError(err)
|
|
|
|
defer s.Close()
|
2022-01-22 04:18:24 +00:00
|
|
|
require.Equal(2, len(s.blocks))
|
2021-12-21 14:12:32 +00:00
|
|
|
|
|
|
|
createFile(500_000, 900_000, Headers)
|
|
|
|
createFile(500_000, 900_000, Bodies)
|
|
|
|
createFile(500_000, 900_000, Transactions)
|
2022-01-20 05:01:02 +00:00
|
|
|
chainSnapshotCfg.ExpectBlocks = math.MaxUint64
|
2022-02-18 02:24:17 +00:00
|
|
|
s = NewRoSnapshots(cfg, dir)
|
2021-12-21 14:12:32 +00:00
|
|
|
defer s.Close()
|
|
|
|
err = s.ReopenSegments()
|
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)
|
2022-02-23 03:54:35 +00:00
|
|
|
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) os.FileInfo {
|
|
|
|
s, err := fs.Stat(name)
|
|
|
|
require.NoError(err)
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
_, err := ParseFileName("", stat("a"))
|
2021-11-16 08:33:41 +00:00
|
|
|
require.Error(err)
|
2022-02-23 03:54:35 +00:00
|
|
|
_, err = ParseFileName("", stat("1-a"))
|
2021-11-16 08:33:41 +00:00
|
|
|
require.Error(err)
|
2022-02-23 03:54:35 +00:00
|
|
|
_, err = ParseFileName("", stat("1-2-a"))
|
2021-11-16 08:33:41 +00:00
|
|
|
require.Error(err)
|
2022-02-23 03:54:35 +00:00
|
|
|
_, err = ParseFileName("", stat("1-2-bodies.info"))
|
2021-11-16 08:33:41 +00:00
|
|
|
require.Error(err)
|
2022-02-23 03:54:35 +00:00
|
|
|
_, err = ParseFileName("", stat("1-2-bodies.seg"))
|
2021-11-16 08:33:41 +00:00
|
|
|
require.Error(err)
|
2022-02-23 03:54:35 +00:00
|
|
|
_, err = ParseFileName("", stat("v2-1-2-bodies.seg"))
|
2021-11-16 08:33:41 +00:00
|
|
|
require.Error(err)
|
2022-02-23 03:54:35 +00:00
|
|
|
_, err = ParseFileName("", stat("v0-1-2-bodies.seg"))
|
2021-11-16 08:33:41 +00:00
|
|
|
require.Error(err)
|
|
|
|
|
2022-02-23 03:54:35 +00:00
|
|
|
f, err := ParseFileName("", stat("v1-1-2-bodies.seg"))
|
2021-11-16 08:33:41 +00:00
|
|
|
require.NoError(err)
|
2022-02-23 03:54:35 +00:00
|
|
|
require.Equal(f.T, Bodies)
|
|
|
|
require.Equal(1_000, int(f.From))
|
|
|
|
require.Equal(2_000, int(f.To))
|
2021-11-16 08:33:41 +00:00
|
|
|
}
|