mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-03 01:27:38 +00:00
64072ce6c7
Co-authored-by: battlmonstr <battlmonstr@users.noreply.github.com>
285 lines
9.0 KiB
Go
285 lines
9.0 KiB
Go
package sync
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"math"
|
|
"math/big"
|
|
"testing"
|
|
|
|
"github.com/golang/mock/gomock"
|
|
"github.com/ledgerwatch/log/v3"
|
|
"github.com/stretchr/testify/require"
|
|
|
|
"github.com/ledgerwatch/erigon-lib/common"
|
|
"github.com/ledgerwatch/erigon/consensus/bor/heimdall/checkpoint"
|
|
"github.com/ledgerwatch/erigon/consensus/bor/heimdall/milestone"
|
|
"github.com/ledgerwatch/erigon/core/types"
|
|
"github.com/ledgerwatch/erigon/polygon/sync/mock"
|
|
"github.com/ledgerwatch/erigon/polygon/sync/peerinfo"
|
|
"github.com/ledgerwatch/erigon/turbo/testlog"
|
|
)
|
|
|
|
func newHeaderDownloaderTest(t *testing.T) *headerDownloaderTest {
|
|
return newHeaderDownloaderTestWithOpts(t, headerDownloaderTestOpts{})
|
|
}
|
|
|
|
func newHeaderDownloaderTestWithOpts(t *testing.T, opts headerDownloaderTestOpts) *headerDownloaderTest {
|
|
ctrl := gomock.NewController(t)
|
|
heimdall := mock.NewMockHeimdall(ctrl)
|
|
sentry := mock.NewMockSentry(ctrl)
|
|
sentry.EXPECT().MaxPeers().Return(100).Times(1)
|
|
db := mock.NewMockDB(ctrl)
|
|
logger := testlog.Logger(t, log.LvlDebug)
|
|
headerVerifier := opts.getOrCreateDefaultHeaderVerifier()
|
|
headerDownloader := NewHeaderDownloader(logger, sentry, db, heimdall, headerVerifier)
|
|
return &headerDownloaderTest{
|
|
heimdall: heimdall,
|
|
sentry: sentry,
|
|
db: db,
|
|
headerDownloader: headerDownloader,
|
|
}
|
|
}
|
|
|
|
type headerDownloaderTestOpts struct {
|
|
headerVerifier HeaderVerifier
|
|
}
|
|
|
|
func (opts headerDownloaderTestOpts) getOrCreateDefaultHeaderVerifier() HeaderVerifier {
|
|
if opts.headerVerifier == nil {
|
|
return func(_ *statePoint, _ []*types.Header) error {
|
|
return nil
|
|
}
|
|
}
|
|
|
|
return opts.headerVerifier
|
|
}
|
|
|
|
type headerDownloaderTest struct {
|
|
heimdall *mock.MockHeimdall
|
|
sentry *mock.MockSentry
|
|
db *mock.MockDB
|
|
headerDownloader *HeaderDownloader
|
|
}
|
|
|
|
func (hdt headerDownloaderTest) fakePeers(count int, blockNums ...*big.Int) peerinfo.PeersWithBlockNumInfo {
|
|
peers := make(peerinfo.PeersWithBlockNumInfo, count)
|
|
for i := range peers {
|
|
var blockNum *big.Int
|
|
if i < len(blockNums) {
|
|
blockNum = blockNums[i]
|
|
} else {
|
|
blockNum = new(big.Int).SetUint64(math.MaxUint64)
|
|
}
|
|
|
|
peers[i] = &peerinfo.PeerWithBlockNumInfo{
|
|
ID: fmt.Sprintf("peer%d", i+1),
|
|
BlockNum: blockNum,
|
|
}
|
|
}
|
|
|
|
return peers
|
|
}
|
|
|
|
func (hdt headerDownloaderTest) fakeCheckpoints(count int) []*checkpoint.Checkpoint {
|
|
checkpoints := make([]*checkpoint.Checkpoint, count)
|
|
for i := range checkpoints {
|
|
num := i + 1
|
|
checkpoints[i] = &checkpoint.Checkpoint{
|
|
StartBlock: big.NewInt(int64(num)),
|
|
EndBlock: big.NewInt(int64(num)),
|
|
RootHash: common.BytesToHash([]byte(fmt.Sprintf("0x%d", num))),
|
|
}
|
|
}
|
|
|
|
return checkpoints
|
|
}
|
|
|
|
func (hdt headerDownloaderTest) fakeMilestones(count int) []*milestone.Milestone {
|
|
milestones := make([]*milestone.Milestone, count)
|
|
for i := range milestones {
|
|
num := i + 1
|
|
milestones[i] = &milestone.Milestone{
|
|
StartBlock: big.NewInt(int64(num)),
|
|
EndBlock: big.NewInt(int64(num)),
|
|
Hash: common.BytesToHash([]byte(fmt.Sprintf("0x%d", num))),
|
|
}
|
|
}
|
|
|
|
return milestones
|
|
}
|
|
|
|
type downloadHeadersMock func(context.Context, *big.Int, *big.Int, string) ([]*types.Header, error)
|
|
|
|
func (hdt headerDownloaderTest) defaultDownloadHeadersMock() downloadHeadersMock {
|
|
return func(ctx context.Context, start *big.Int, end *big.Int, peerID string) ([]*types.Header, error) {
|
|
res := make([]*types.Header, new(big.Int).Sub(end, start).Uint64()+1)
|
|
for i := new(big.Int).Set(start); i.Cmp(end) < 1; i.Add(i, new(big.Int).SetUint64(1)) {
|
|
res[new(big.Int).Sub(i, start).Uint64()] = &types.Header{Number: new(big.Int).Set(i)}
|
|
}
|
|
return res, nil
|
|
}
|
|
}
|
|
|
|
func (hdt headerDownloaderTest) defaultWriteHeadersMock(capture *[]*types.Header) func([]*types.Header) error {
|
|
return func(headers []*types.Header) error {
|
|
*capture = append(*capture, headers...)
|
|
return nil
|
|
}
|
|
}
|
|
|
|
func TestHeaderDownloadUsingMilestones(t *testing.T) {
|
|
test := newHeaderDownloaderTest(t)
|
|
test.heimdall.EXPECT().
|
|
FetchMilestones(gomock.Any(), gomock.Any()).
|
|
Return(test.fakeMilestones(4), nil).
|
|
Times(1)
|
|
test.sentry.EXPECT().
|
|
PeersWithBlockNumInfo().
|
|
Return(test.fakePeers(8)).
|
|
Times(1)
|
|
test.sentry.EXPECT().
|
|
DownloadHeaders(gomock.Any(), gomock.Any(), gomock.Any(), gomock.Any()).
|
|
DoAndReturn(test.defaultDownloadHeadersMock()).
|
|
Times(4)
|
|
var persistedHeaders []*types.Header
|
|
test.db.EXPECT().
|
|
WriteHeaders(gomock.Any()).
|
|
DoAndReturn(test.defaultWriteHeadersMock(&persistedHeaders)).
|
|
Times(1)
|
|
|
|
err := test.headerDownloader.DownloadUsingMilestones(context.Background(), 1)
|
|
require.NoError(t, err)
|
|
require.Len(t, persistedHeaders, 4)
|
|
// check headers are written in order
|
|
require.Equal(t, uint64(1), persistedHeaders[0].Number.Uint64())
|
|
require.Equal(t, uint64(2), persistedHeaders[1].Number.Uint64())
|
|
require.Equal(t, uint64(3), persistedHeaders[2].Number.Uint64())
|
|
require.Equal(t, uint64(4), persistedHeaders[3].Number.Uint64())
|
|
}
|
|
|
|
func TestHeaderDownloadUsingCheckpoints(t *testing.T) {
|
|
test := newHeaderDownloaderTest(t)
|
|
test.heimdall.EXPECT().
|
|
FetchCheckpoints(gomock.Any(), gomock.Any()).
|
|
Return(test.fakeCheckpoints(8), nil).
|
|
Times(1)
|
|
test.sentry.EXPECT().
|
|
PeersWithBlockNumInfo().
|
|
Return(test.fakePeers(2)).
|
|
Times(4)
|
|
test.sentry.EXPECT().
|
|
DownloadHeaders(gomock.Any(), gomock.Any(), gomock.Any(), gomock.Any()).
|
|
DoAndReturn(test.defaultDownloadHeadersMock()).
|
|
Times(8)
|
|
var persistedHeaders []*types.Header
|
|
test.db.EXPECT().
|
|
WriteHeaders(gomock.Any()).
|
|
DoAndReturn(test.defaultWriteHeadersMock(&persistedHeaders)).
|
|
Times(4)
|
|
|
|
err := test.headerDownloader.DownloadUsingCheckpoints(context.Background(), 1)
|
|
require.NoError(t, err)
|
|
require.Len(t, persistedHeaders, 8)
|
|
// check headers are written in order
|
|
require.Equal(t, uint64(1), persistedHeaders[0].Number.Uint64())
|
|
require.Equal(t, uint64(2), persistedHeaders[1].Number.Uint64())
|
|
require.Equal(t, uint64(3), persistedHeaders[2].Number.Uint64())
|
|
require.Equal(t, uint64(4), persistedHeaders[3].Number.Uint64())
|
|
require.Equal(t, uint64(5), persistedHeaders[4].Number.Uint64())
|
|
require.Equal(t, uint64(6), persistedHeaders[5].Number.Uint64())
|
|
require.Equal(t, uint64(7), persistedHeaders[6].Number.Uint64())
|
|
require.Equal(t, uint64(8), persistedHeaders[7].Number.Uint64())
|
|
}
|
|
|
|
func TestHeaderDownloadWhenInvalidStateThenPenalizePeerAndReDownload(t *testing.T) {
|
|
var firstTimeInvalidReturned bool
|
|
firstTimeInvalidReturnedPtr := &firstTimeInvalidReturned
|
|
test := newHeaderDownloaderTestWithOpts(t, headerDownloaderTestOpts{
|
|
headerVerifier: func(statePoint *statePoint, headers []*types.Header) error {
|
|
if statePoint.startBlock.Cmp(new(big.Int).SetUint64(2)) == 0 && !*firstTimeInvalidReturnedPtr {
|
|
*firstTimeInvalidReturnedPtr = true
|
|
return errors.New("invalid checkpoint")
|
|
}
|
|
return nil
|
|
},
|
|
})
|
|
test.heimdall.EXPECT().
|
|
FetchCheckpoints(gomock.Any(), gomock.Any()).
|
|
Return(test.fakeCheckpoints(6), nil).
|
|
Times(1)
|
|
test.sentry.EXPECT().
|
|
PeersWithBlockNumInfo().
|
|
Return(test.fakePeers(3)).
|
|
Times(3)
|
|
test.sentry.EXPECT().
|
|
DownloadHeaders(gomock.Any(), gomock.Any(), gomock.Any(), gomock.Any()).
|
|
DoAndReturn(test.defaultDownloadHeadersMock()).
|
|
// request 1,2,3 in parallel
|
|
// -> 2 fails
|
|
// requests 2,3,4 in parallel
|
|
// 3 is cached
|
|
// requests 5,6 in parallel
|
|
// in total 6 requests + 1 request for re-requesting checkpoint 2
|
|
// total = 7 (note this also tests caching works)
|
|
Times(7)
|
|
test.sentry.EXPECT().
|
|
Penalize(gomock.Eq("peer2")).
|
|
Times(1)
|
|
var persistedHeadersFirstTime, persistedHeadersRemaining []*types.Header
|
|
gomock.InOrder(
|
|
test.db.EXPECT().
|
|
WriteHeaders(gomock.Any()).
|
|
DoAndReturn(test.defaultWriteHeadersMock(&persistedHeadersFirstTime)).
|
|
Times(1),
|
|
test.db.EXPECT().
|
|
WriteHeaders(gomock.Any()).
|
|
DoAndReturn(test.defaultWriteHeadersMock(&persistedHeadersRemaining)).
|
|
Times(2),
|
|
)
|
|
|
|
err := test.headerDownloader.DownloadUsingCheckpoints(context.Background(), 1)
|
|
require.NoError(t, err)
|
|
require.Len(t, persistedHeadersFirstTime, 1)
|
|
require.Len(t, persistedHeadersRemaining, 5)
|
|
}
|
|
|
|
func TestHeaderDownloadWhenZeroPeersTriesAgain(t *testing.T) {
|
|
test := newHeaderDownloaderTest(t)
|
|
test.heimdall.EXPECT().
|
|
FetchCheckpoints(gomock.Any(), gomock.Any()).
|
|
Return(test.fakeCheckpoints(8), nil).
|
|
Times(1)
|
|
test.sentry.EXPECT().
|
|
DownloadHeaders(gomock.Any(), gomock.Any(), gomock.Any(), gomock.Any()).
|
|
DoAndReturn(test.defaultDownloadHeadersMock()).
|
|
Times(8)
|
|
var persistedHeaders []*types.Header
|
|
test.db.EXPECT().
|
|
WriteHeaders(gomock.Any()).
|
|
DoAndReturn(test.defaultWriteHeadersMock(&persistedHeaders)).
|
|
Times(4)
|
|
gomock.InOrder(
|
|
// first, no peers at all
|
|
test.sentry.EXPECT().
|
|
PeersWithBlockNumInfo().
|
|
Return(nil).
|
|
Times(1),
|
|
// second, 2 peers but not synced enough for us to use
|
|
test.sentry.EXPECT().
|
|
PeersWithBlockNumInfo().
|
|
Return(test.fakePeers(2, new(big.Int).SetUint64(0), new(big.Int).SetUint64(0))).
|
|
Times(1),
|
|
// then, 2 fully synced peers that we can use
|
|
test.sentry.EXPECT().
|
|
PeersWithBlockNumInfo().
|
|
Return(test.fakePeers(2)).
|
|
Times(4),
|
|
)
|
|
|
|
err := test.headerDownloader.DownloadUsingCheckpoints(context.Background(), 1)
|
|
require.NoError(t, err)
|
|
require.Len(t, persistedHeaders, 8)
|
|
}
|