mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-14 05:58:19 +00:00
767 lines
25 KiB
Go
767 lines
25 KiB
Go
package powchain
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"math/big"
|
|
"strings"
|
|
"sync"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/ethereum/go-ethereum"
|
|
"github.com/ethereum/go-ethereum/accounts/abi/bind/backends"
|
|
"github.com/ethereum/go-ethereum/common"
|
|
gethTypes "github.com/ethereum/go-ethereum/core/types"
|
|
ethpb "github.com/prysmaticlabs/ethereumapis/eth/v1alpha1"
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/cache/depositcache"
|
|
dbutil "github.com/prysmaticlabs/prysm/beacon-chain/db/testing"
|
|
mockPOW "github.com/prysmaticlabs/prysm/beacon-chain/powchain/testing"
|
|
contracts "github.com/prysmaticlabs/prysm/contracts/deposit-contract"
|
|
protodb "github.com/prysmaticlabs/prysm/proto/beacon/db"
|
|
"github.com/prysmaticlabs/prysm/shared/clientstats"
|
|
"github.com/prysmaticlabs/prysm/shared/event"
|
|
"github.com/prysmaticlabs/prysm/shared/httputils"
|
|
"github.com/prysmaticlabs/prysm/shared/params"
|
|
"github.com/prysmaticlabs/prysm/shared/testutil"
|
|
"github.com/prysmaticlabs/prysm/shared/testutil/assert"
|
|
"github.com/prysmaticlabs/prysm/shared/testutil/require"
|
|
logTest "github.com/sirupsen/logrus/hooks/test"
|
|
)
|
|
|
|
var _ ChainStartFetcher = (*Service)(nil)
|
|
var _ ChainInfoFetcher = (*Service)(nil)
|
|
var _ POWBlockFetcher = (*Service)(nil)
|
|
var _ Chain = (*Service)(nil)
|
|
|
|
type goodLogger struct {
|
|
backend *backends.SimulatedBackend
|
|
}
|
|
|
|
func (g *goodLogger) SubscribeFilterLogs(ctx context.Context, q ethereum.FilterQuery, ch chan<- gethTypes.Log) (ethereum.Subscription, error) {
|
|
if g.backend == nil {
|
|
return new(event.Feed).Subscribe(ch), nil
|
|
}
|
|
return g.backend.SubscribeFilterLogs(ctx, q, ch)
|
|
}
|
|
|
|
func (g *goodLogger) FilterLogs(ctx context.Context, q ethereum.FilterQuery) ([]gethTypes.Log, error) {
|
|
if g.backend == nil {
|
|
logs := make([]gethTypes.Log, 3)
|
|
for i := 0; i < len(logs); i++ {
|
|
logs[i].Address = common.Address{}
|
|
logs[i].Topics = make([]common.Hash, 5)
|
|
logs[i].Topics[0] = common.Hash{'a'}
|
|
logs[i].Topics[1] = common.Hash{'b'}
|
|
logs[i].Topics[2] = common.Hash{'c'}
|
|
|
|
}
|
|
return logs, nil
|
|
}
|
|
return g.backend.FilterLogs(ctx, q)
|
|
}
|
|
|
|
type goodNotifier struct {
|
|
MockStateFeed *event.Feed
|
|
}
|
|
|
|
func (g *goodNotifier) StateFeed() *event.Feed {
|
|
if g.MockStateFeed == nil {
|
|
g.MockStateFeed = new(event.Feed)
|
|
}
|
|
return g.MockStateFeed
|
|
}
|
|
|
|
type goodFetcher struct {
|
|
backend *backends.SimulatedBackend
|
|
}
|
|
|
|
func (g *goodFetcher) HeaderByHash(_ context.Context, hash common.Hash) (*gethTypes.Header, error) {
|
|
if bytes.Equal(hash.Bytes(), common.BytesToHash([]byte{0}).Bytes()) {
|
|
return nil, fmt.Errorf("expected block hash to be nonzero %v", hash)
|
|
}
|
|
if g.backend == nil {
|
|
return &gethTypes.Header{
|
|
Number: big.NewInt(0),
|
|
}, nil
|
|
}
|
|
header := g.backend.Blockchain().GetHeaderByHash(hash)
|
|
if header == nil {
|
|
return nil, errors.New("nil header returned")
|
|
}
|
|
return header, nil
|
|
|
|
}
|
|
|
|
func (g *goodFetcher) HeaderByNumber(_ context.Context, number *big.Int) (*gethTypes.Header, error) {
|
|
if g.backend == nil {
|
|
return &gethTypes.Header{
|
|
Number: big.NewInt(15),
|
|
Time: 150,
|
|
}, nil
|
|
}
|
|
var header *gethTypes.Header
|
|
if number == nil {
|
|
header = g.backend.Blockchain().CurrentHeader()
|
|
} else {
|
|
header = g.backend.Blockchain().GetHeaderByNumber(number.Uint64())
|
|
}
|
|
if header == nil {
|
|
return nil, errors.New("nil header returned")
|
|
}
|
|
return header, nil
|
|
}
|
|
|
|
func (g *goodFetcher) SyncProgress(_ context.Context) (*ethereum.SyncProgress, error) {
|
|
return nil, nil
|
|
}
|
|
|
|
var depositsReqForChainStart = 64
|
|
|
|
func TestStart_OK(t *testing.T) {
|
|
hook := logTest.NewGlobal()
|
|
beaconDB := dbutil.SetupDB(t)
|
|
testAcc, err := contracts.Setup()
|
|
require.NoError(t, err, "Unable to set up simulated backend")
|
|
web3Service, err := NewService(context.Background(), &Web3ServiceConfig{
|
|
HttpEndpoints: []string{endpoint},
|
|
DepositContract: testAcc.ContractAddr,
|
|
BeaconDB: beaconDB,
|
|
})
|
|
require.NoError(t, err, "unable to setup web3 ETH1.0 chain service")
|
|
web3Service = setDefaultMocks(web3Service)
|
|
web3Service.rpcClient = &mockPOW.RPCClient{Backend: testAcc.Backend}
|
|
web3Service.depositContractCaller, err = contracts.NewDepositContractCaller(testAcc.ContractAddr, testAcc.Backend)
|
|
require.NoError(t, err)
|
|
testAcc.Backend.Commit()
|
|
|
|
web3Service.Start()
|
|
if len(hook.Entries) > 0 {
|
|
msg := hook.LastEntry().Message
|
|
want := "Could not connect to ETH1.0 chain RPC client"
|
|
if strings.Contains(want, msg) {
|
|
t.Errorf("incorrect log, expected %s, got %s", want, msg)
|
|
}
|
|
}
|
|
hook.Reset()
|
|
web3Service.cancel()
|
|
}
|
|
|
|
func TestStart_NoHttpEndpointDefinedFails_WithoutChainStarted(t *testing.T) {
|
|
hook := logTest.NewGlobal()
|
|
beaconDB := dbutil.SetupDB(t)
|
|
testAcc, err := contracts.Setup()
|
|
require.NoError(t, err, "Unable to set up simulated backend")
|
|
s, err := NewService(context.Background(), &Web3ServiceConfig{
|
|
HttpEndpoints: []string{""}, // No endpoint defined!
|
|
DepositContract: testAcc.ContractAddr,
|
|
BeaconDB: beaconDB,
|
|
})
|
|
require.NoError(t, err)
|
|
// Set custom exit func so test can proceed
|
|
log.Logger.ExitFunc = func(i int) {
|
|
panic(i)
|
|
}
|
|
defer func() {
|
|
log.Logger.ExitFunc = nil
|
|
}()
|
|
wg := new(sync.WaitGroup)
|
|
wg.Add(1)
|
|
// Expect Start function to fail from a fatal call due
|
|
// to no state existing.
|
|
go func() {
|
|
defer func() {
|
|
if r := recover(); r != nil {
|
|
wg.Done()
|
|
}
|
|
}()
|
|
s.Start()
|
|
}()
|
|
testutil.WaitTimeout(wg, time.Second)
|
|
require.LogsContain(t, hook, "cannot create genesis state: no eth1 http endpoint defined")
|
|
hook.Reset()
|
|
}
|
|
|
|
func TestStart_NoHttpEndpointDefinedSucceeds_WithGenesisState(t *testing.T) {
|
|
hook := logTest.NewGlobal()
|
|
beaconDB := dbutil.SetupDB(t)
|
|
testAcc, err := contracts.Setup()
|
|
require.NoError(t, err, "Unable to set up simulated backend")
|
|
st, _ := testutil.DeterministicGenesisState(t, 10)
|
|
b := testutil.NewBeaconBlock()
|
|
genRoot, err := b.HashTreeRoot()
|
|
require.NoError(t, err)
|
|
require.NoError(t, beaconDB.SaveState(context.Background(), st, genRoot))
|
|
require.NoError(t, beaconDB.SaveGenesisBlockRoot(context.Background(), genRoot))
|
|
depositCache, err := depositcache.New()
|
|
require.NoError(t, err)
|
|
s, err := NewService(context.Background(), &Web3ServiceConfig{
|
|
HttpEndpoints: []string{""}, // No endpoint defined!
|
|
DepositContract: testAcc.ContractAddr,
|
|
BeaconDB: beaconDB,
|
|
DepositCache: depositCache,
|
|
})
|
|
require.NoError(t, err)
|
|
|
|
wg := new(sync.WaitGroup)
|
|
wg.Add(1)
|
|
|
|
go func() {
|
|
s.Start()
|
|
wg.Done()
|
|
}()
|
|
s.cancel()
|
|
testutil.WaitTimeout(wg, time.Second)
|
|
require.LogsDoNotContain(t, hook, "cannot create genesis state: no eth1 http endpoint defined")
|
|
hook.Reset()
|
|
}
|
|
|
|
func TestStart_NoHttpEndpointDefinedSucceeds_WithChainStarted(t *testing.T) {
|
|
hook := logTest.NewGlobal()
|
|
beaconDB := dbutil.SetupDB(t)
|
|
testAcc, err := contracts.Setup()
|
|
require.NoError(t, err, "Unable to set up simulated backend")
|
|
|
|
require.NoError(t, beaconDB.SavePowchainData(context.Background(), &protodb.ETH1ChainData{
|
|
ChainstartData: &protodb.ChainStartData{Chainstarted: true},
|
|
Trie: &protodb.SparseMerkleTrie{},
|
|
}))
|
|
s, err := NewService(context.Background(), &Web3ServiceConfig{
|
|
HttpEndpoints: []string{""}, // No endpoint defined!
|
|
DepositContract: testAcc.ContractAddr,
|
|
BeaconDB: beaconDB,
|
|
})
|
|
require.NoError(t, err)
|
|
|
|
s.Start()
|
|
require.LogsDoNotContain(t, hook, "cannot create genesis state: no eth1 http endpoint defined")
|
|
hook.Reset()
|
|
}
|
|
|
|
func TestStop_OK(t *testing.T) {
|
|
hook := logTest.NewGlobal()
|
|
testAcc, err := contracts.Setup()
|
|
require.NoError(t, err, "Unable to set up simulated backend")
|
|
beaconDB := dbutil.SetupDB(t)
|
|
web3Service, err := NewService(context.Background(), &Web3ServiceConfig{
|
|
HttpEndpoints: []string{endpoint},
|
|
DepositContract: testAcc.ContractAddr,
|
|
BeaconDB: beaconDB,
|
|
})
|
|
require.NoError(t, err, "unable to setup web3 ETH1.0 chain service")
|
|
web3Service = setDefaultMocks(web3Service)
|
|
web3Service.depositContractCaller, err = contracts.NewDepositContractCaller(testAcc.ContractAddr, testAcc.Backend)
|
|
require.NoError(t, err)
|
|
|
|
testAcc.Backend.Commit()
|
|
|
|
err = web3Service.Stop()
|
|
require.NoError(t, err, "Unable to stop web3 ETH1.0 chain service")
|
|
|
|
// The context should have been canceled.
|
|
assert.NotNil(t, web3Service.ctx.Err(), "Context wasnt canceled")
|
|
|
|
hook.Reset()
|
|
}
|
|
|
|
func TestService_Eth1Synced(t *testing.T) {
|
|
testAcc, err := contracts.Setup()
|
|
require.NoError(t, err, "Unable to set up simulated backend")
|
|
beaconDB := dbutil.SetupDB(t)
|
|
web3Service, err := NewService(context.Background(), &Web3ServiceConfig{
|
|
HttpEndpoints: []string{endpoint},
|
|
DepositContract: testAcc.ContractAddr,
|
|
BeaconDB: beaconDB,
|
|
})
|
|
require.NoError(t, err, "unable to setup web3 ETH1.0 chain service")
|
|
web3Service = setDefaultMocks(web3Service)
|
|
web3Service.depositContractCaller, err = contracts.NewDepositContractCaller(testAcc.ContractAddr, testAcc.Backend)
|
|
require.NoError(t, err)
|
|
web3Service.eth1DataFetcher = &goodFetcher{backend: testAcc.Backend}
|
|
|
|
currTime := testAcc.Backend.Blockchain().CurrentHeader().Time
|
|
now := time.Now()
|
|
assert.NoError(t, testAcc.Backend.AdjustTime(now.Sub(time.Unix(int64(currTime), 0))))
|
|
testAcc.Backend.Commit()
|
|
|
|
synced, err := web3Service.isEth1NodeSynced()
|
|
require.NoError(t, err)
|
|
assert.Equal(t, true, synced, "Expected eth1 nodes to be synced")
|
|
}
|
|
|
|
func TestFollowBlock_OK(t *testing.T) {
|
|
testAcc, err := contracts.Setup()
|
|
require.NoError(t, err, "Unable to set up simulated backend")
|
|
beaconDB := dbutil.SetupDB(t)
|
|
web3Service, err := NewService(context.Background(), &Web3ServiceConfig{
|
|
HttpEndpoints: []string{endpoint},
|
|
DepositContract: testAcc.ContractAddr,
|
|
BeaconDB: beaconDB,
|
|
})
|
|
require.NoError(t, err, "unable to setup web3 ETH1.0 chain service")
|
|
|
|
// simulated backend sets eth1 block
|
|
// time as 10 seconds
|
|
conf := params.BeaconConfig()
|
|
conf.SecondsPerETH1Block = 10
|
|
params.OverrideBeaconConfig(conf)
|
|
defer params.UseMainnetConfig()
|
|
|
|
web3Service = setDefaultMocks(web3Service)
|
|
web3Service.eth1DataFetcher = &goodFetcher{backend: testAcc.Backend}
|
|
baseHeight := testAcc.Backend.Blockchain().CurrentBlock().NumberU64()
|
|
// process follow_distance blocks
|
|
for i := 0; i < int(params.BeaconConfig().Eth1FollowDistance); i++ {
|
|
testAcc.Backend.Commit()
|
|
}
|
|
// set current height
|
|
web3Service.latestEth1Data.BlockHeight = testAcc.Backend.Blockchain().CurrentBlock().NumberU64()
|
|
web3Service.latestEth1Data.BlockTime = testAcc.Backend.Blockchain().CurrentBlock().Time()
|
|
|
|
h, err := web3Service.followBlockHeight(context.Background())
|
|
require.NoError(t, err)
|
|
assert.Equal(t, baseHeight, h, "Unexpected block height")
|
|
numToForward := uint64(2)
|
|
expectedHeight := numToForward + baseHeight
|
|
// forward 2 blocks
|
|
for i := uint64(0); i < numToForward; i++ {
|
|
testAcc.Backend.Commit()
|
|
}
|
|
// set current height
|
|
web3Service.latestEth1Data.BlockHeight = testAcc.Backend.Blockchain().CurrentBlock().NumberU64()
|
|
web3Service.latestEth1Data.BlockTime = testAcc.Backend.Blockchain().CurrentBlock().Time()
|
|
|
|
h, err = web3Service.followBlockHeight(context.Background())
|
|
require.NoError(t, err)
|
|
assert.Equal(t, expectedHeight, h, "Unexpected block height")
|
|
}
|
|
|
|
func TestStatus(t *testing.T) {
|
|
now := time.Now()
|
|
|
|
beforeFiveMinutesAgo := uint64(now.Add(-5*time.Minute - 30*time.Second).Unix())
|
|
afterFiveMinutesAgo := uint64(now.Add(-5*time.Minute + 30*time.Second).Unix())
|
|
|
|
testCases := map[*Service]string{
|
|
// "status is ok" cases
|
|
{}: "",
|
|
{isRunning: true, latestEth1Data: &protodb.LatestETH1Data{BlockTime: afterFiveMinutesAgo}}: "",
|
|
{isRunning: false, latestEth1Data: &protodb.LatestETH1Data{BlockTime: beforeFiveMinutesAgo}}: "",
|
|
{isRunning: false, runError: errors.New("test runError")}: "",
|
|
// "status is error" cases
|
|
{isRunning: true, runError: errors.New("test runError")}: "test runError",
|
|
}
|
|
|
|
for web3ServiceState, wantedErrorText := range testCases {
|
|
status := web3ServiceState.Status()
|
|
if status == nil {
|
|
assert.Equal(t, "", wantedErrorText)
|
|
|
|
} else {
|
|
assert.Equal(t, wantedErrorText, status.Error())
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestHandlePanic_OK(t *testing.T) {
|
|
hook := logTest.NewGlobal()
|
|
beaconDB := dbutil.SetupDB(t)
|
|
web3Service, err := NewService(context.Background(), &Web3ServiceConfig{
|
|
HttpEndpoints: []string{endpoint},
|
|
BeaconDB: beaconDB,
|
|
})
|
|
require.NoError(t, err, "unable to setup web3 ETH1.0 chain service")
|
|
// nil eth1DataFetcher would panic if cached value not used
|
|
web3Service.eth1DataFetcher = nil
|
|
web3Service.processBlockHeader(nil)
|
|
require.LogsContain(t, hook, "Panicked when handling data from ETH 1.0 Chain!")
|
|
}
|
|
|
|
func TestLogTillGenesis_OK(t *testing.T) {
|
|
// Reset the var at the end of the test.
|
|
currPeriod := logPeriod
|
|
logPeriod = 1 * time.Second
|
|
defer func() {
|
|
logPeriod = currPeriod
|
|
}()
|
|
|
|
orgConfig := params.BeaconConfig().Copy()
|
|
cfg := params.BeaconConfig()
|
|
cfg.Eth1FollowDistance = 5
|
|
params.OverrideBeaconConfig(cfg)
|
|
defer func() {
|
|
params.OverrideBeaconConfig(orgConfig)
|
|
}()
|
|
|
|
orgNetworkConfig := params.BeaconNetworkConfig().Copy()
|
|
nCfg := params.BeaconNetworkConfig()
|
|
nCfg.ContractDeploymentBlock = 0
|
|
params.OverrideBeaconNetworkConfig(nCfg)
|
|
defer func() {
|
|
params.OverrideBeaconNetworkConfig(orgNetworkConfig)
|
|
}()
|
|
|
|
hook := logTest.NewGlobal()
|
|
testAcc, err := contracts.Setup()
|
|
require.NoError(t, err, "Unable to set up simulated backend")
|
|
beaconDB := dbutil.SetupDB(t)
|
|
web3Service, err := NewService(context.Background(), &Web3ServiceConfig{
|
|
HttpEndpoints: []string{endpoint},
|
|
DepositContract: testAcc.ContractAddr,
|
|
BeaconDB: beaconDB,
|
|
})
|
|
require.NoError(t, err, "unable to setup web3 ETH1.0 chain service")
|
|
web3Service.depositContractCaller, err = contracts.NewDepositContractCaller(testAcc.ContractAddr, testAcc.Backend)
|
|
require.NoError(t, err)
|
|
|
|
web3Service.rpcClient = &mockPOW.RPCClient{Backend: testAcc.Backend}
|
|
web3Service.eth1DataFetcher = &goodFetcher{backend: testAcc.Backend}
|
|
web3Service.httpLogger = testAcc.Backend
|
|
for i := 0; i < 30; i++ {
|
|
testAcc.Backend.Commit()
|
|
}
|
|
web3Service.latestEth1Data = &protodb.LatestETH1Data{LastRequestedBlock: 0}
|
|
// Spin off to a separate routine
|
|
go web3Service.run(web3Service.ctx.Done())
|
|
// Wait for 2 seconds so that the
|
|
// info is logged.
|
|
time.Sleep(2 * time.Second)
|
|
web3Service.cancel()
|
|
assert.LogsContain(t, hook, "Currently waiting for chainstart")
|
|
}
|
|
|
|
func TestInitDepositCache_OK(t *testing.T) {
|
|
ctrs := []*protodb.DepositContainer{
|
|
{Index: 0, Eth1BlockHeight: 2, Deposit: ðpb.Deposit{Proof: [][]byte{[]byte("A")}}},
|
|
{Index: 1, Eth1BlockHeight: 4, Deposit: ðpb.Deposit{Proof: [][]byte{[]byte("B")}}},
|
|
{Index: 2, Eth1BlockHeight: 6, Deposit: ðpb.Deposit{Proof: [][]byte{[]byte("c")}}},
|
|
}
|
|
gs, _ := testutil.DeterministicGenesisState(t, 1)
|
|
beaconDB := dbutil.SetupDB(t)
|
|
s := &Service{
|
|
chainStartData: &protodb.ChainStartData{Chainstarted: false},
|
|
preGenesisState: gs,
|
|
cfg: &Web3ServiceConfig{BeaconDB: beaconDB},
|
|
}
|
|
var err error
|
|
s.cfg.DepositCache, err = depositcache.New()
|
|
require.NoError(t, err)
|
|
require.NoError(t, s.initDepositCaches(context.Background(), ctrs))
|
|
|
|
require.Equal(t, 0, len(s.cfg.DepositCache.PendingContainers(context.Background(), nil)))
|
|
|
|
blockRootA := [32]byte{'a'}
|
|
|
|
emptyState, err := testutil.NewBeaconState()
|
|
require.NoError(t, err)
|
|
require.NoError(t, s.cfg.BeaconDB.SaveGenesisBlockRoot(context.Background(), blockRootA))
|
|
require.NoError(t, s.cfg.BeaconDB.SaveState(context.Background(), emptyState, blockRootA))
|
|
s.chainStartData.Chainstarted = true
|
|
require.NoError(t, s.initDepositCaches(context.Background(), ctrs))
|
|
require.Equal(t, 3, len(s.cfg.DepositCache.PendingContainers(context.Background(), nil)))
|
|
}
|
|
|
|
func TestNewService_EarliestVotingBlock(t *testing.T) {
|
|
testAcc, err := contracts.Setup()
|
|
require.NoError(t, err, "Unable to set up simulated backend")
|
|
beaconDB := dbutil.SetupDB(t)
|
|
web3Service, err := NewService(context.Background(), &Web3ServiceConfig{
|
|
HttpEndpoints: []string{endpoint},
|
|
DepositContract: testAcc.ContractAddr,
|
|
BeaconDB: beaconDB,
|
|
})
|
|
require.NoError(t, err, "unable to setup web3 ETH1.0 chain service")
|
|
web3Service.eth1DataFetcher = &goodFetcher{backend: testAcc.Backend}
|
|
// simulated backend sets eth1 block
|
|
// time as 10 seconds
|
|
conf := params.BeaconConfig()
|
|
conf.SecondsPerETH1Block = 10
|
|
conf.Eth1FollowDistance = 50
|
|
params.OverrideBeaconConfig(conf)
|
|
defer params.UseMainnetConfig()
|
|
|
|
// Genesis not set
|
|
followBlock := uint64(2000)
|
|
blk, err := web3Service.determineEarliestVotingBlock(context.Background(), followBlock)
|
|
require.NoError(t, err)
|
|
assert.Equal(t, followBlock-conf.Eth1FollowDistance, blk, "unexpected earliest voting block")
|
|
|
|
// Genesis is set.
|
|
|
|
numToForward := 1500
|
|
// forward 1500 blocks
|
|
for i := 0; i < numToForward; i++ {
|
|
testAcc.Backend.Commit()
|
|
}
|
|
currTime := testAcc.Backend.Blockchain().CurrentHeader().Time
|
|
now := time.Now()
|
|
err = testAcc.Backend.AdjustTime(now.Sub(time.Unix(int64(currTime), 0)))
|
|
require.NoError(t, err)
|
|
testAcc.Backend.Commit()
|
|
|
|
currTime = testAcc.Backend.Blockchain().CurrentHeader().Time
|
|
web3Service.latestEth1Data.BlockHeight = testAcc.Backend.Blockchain().CurrentHeader().Number.Uint64()
|
|
web3Service.latestEth1Data.BlockTime = testAcc.Backend.Blockchain().CurrentHeader().Time
|
|
web3Service.chainStartData.GenesisTime = currTime
|
|
|
|
// With a current slot of zero, only request follow_blocks behind.
|
|
blk, err = web3Service.determineEarliestVotingBlock(context.Background(), followBlock)
|
|
require.NoError(t, err)
|
|
assert.Equal(t, followBlock-conf.Eth1FollowDistance, blk, "unexpected earliest voting block")
|
|
|
|
}
|
|
|
|
func TestNewService_Eth1HeaderRequLimit(t *testing.T) {
|
|
testAcc, err := contracts.Setup()
|
|
require.NoError(t, err, "Unable to set up simulated backend")
|
|
beaconDB := dbutil.SetupDB(t)
|
|
|
|
s1, err := NewService(context.Background(), &Web3ServiceConfig{
|
|
HttpEndpoints: []string{endpoint},
|
|
DepositContract: testAcc.ContractAddr,
|
|
BeaconDB: beaconDB,
|
|
})
|
|
require.NoError(t, err, "unable to setup web3 ETH1.0 chain service")
|
|
assert.Equal(t, defaultEth1HeaderReqLimit, s1.cfg.Eth1HeaderReqLimit, "default eth1 header request limit not set")
|
|
|
|
s2, err := NewService(context.Background(), &Web3ServiceConfig{
|
|
HttpEndpoints: []string{endpoint},
|
|
DepositContract: testAcc.ContractAddr,
|
|
BeaconDB: beaconDB,
|
|
Eth1HeaderReqLimit: uint64(150),
|
|
})
|
|
require.NoError(t, err, "unable to setup web3 ETH1.0 chain service")
|
|
assert.Equal(t, uint64(150), s2.cfg.Eth1HeaderReqLimit, "unable to set eth1HeaderRequestLimit")
|
|
}
|
|
|
|
type mockBSUpdater struct {
|
|
lastBS clientstats.BeaconNodeStats
|
|
}
|
|
|
|
func (mbs *mockBSUpdater) Update(bs clientstats.BeaconNodeStats) {
|
|
mbs.lastBS = bs
|
|
}
|
|
|
|
var _ BeaconNodeStatsUpdater = &mockBSUpdater{}
|
|
|
|
func TestServiceFallbackCorrectly(t *testing.T) {
|
|
firstEndpoint := "A"
|
|
secondEndpoint := "B"
|
|
|
|
testAcc, err := contracts.Setup()
|
|
require.NoError(t, err, "Unable to set up simulated backend")
|
|
beaconDB := dbutil.SetupDB(t)
|
|
|
|
mbs := &mockBSUpdater{}
|
|
s1, err := NewService(context.Background(), &Web3ServiceConfig{
|
|
HttpEndpoints: []string{firstEndpoint},
|
|
DepositContract: testAcc.ContractAddr,
|
|
BeaconDB: beaconDB,
|
|
BeaconNodeStatsUpdater: mbs,
|
|
})
|
|
s1.bsUpdater = mbs
|
|
require.NoError(t, err)
|
|
|
|
assert.Equal(t, firstEndpoint, s1.currHttpEndpoint.Url, "Unexpected http endpoint")
|
|
// Stay at the first endpoint.
|
|
s1.fallbackToNextEndpoint()
|
|
assert.Equal(t, firstEndpoint, s1.currHttpEndpoint.Url, "Unexpected http endpoint")
|
|
assert.Equal(t, false, mbs.lastBS.SyncEth1FallbackConfigured, "SyncEth1FallbackConfigured in clientstats update should be false when only 1 endpoint is configured")
|
|
|
|
s1.httpEndpoints = append(s1.httpEndpoints, httputils.Endpoint{Url: secondEndpoint})
|
|
|
|
s1.fallbackToNextEndpoint()
|
|
assert.Equal(t, secondEndpoint, s1.currHttpEndpoint.Url, "Unexpected http endpoint")
|
|
assert.Equal(t, true, mbs.lastBS.SyncEth1FallbackConfigured, "SyncEth1FallbackConfigured in clientstats update should be true when > 1 endpoint is configured")
|
|
|
|
thirdEndpoint := "C"
|
|
fourthEndpoint := "D"
|
|
|
|
s1.httpEndpoints = append(s1.httpEndpoints, httputils.Endpoint{Url: thirdEndpoint}, httputils.Endpoint{Url: fourthEndpoint})
|
|
|
|
s1.fallbackToNextEndpoint()
|
|
assert.Equal(t, thirdEndpoint, s1.currHttpEndpoint.Url, "Unexpected http endpoint")
|
|
|
|
s1.fallbackToNextEndpoint()
|
|
assert.Equal(t, fourthEndpoint, s1.currHttpEndpoint.Url, "Unexpected http endpoint")
|
|
|
|
// Rollover correctly back to the first endpoint
|
|
s1.fallbackToNextEndpoint()
|
|
assert.Equal(t, firstEndpoint, s1.currHttpEndpoint.Url, "Unexpected http endpoint")
|
|
}
|
|
|
|
func TestDedupEndpoints(t *testing.T) {
|
|
assert.DeepEqual(t, []string{"A"}, dedupEndpoints([]string{"A"}), "did not dedup correctly")
|
|
assert.DeepEqual(t, []string{"A", "B"}, dedupEndpoints([]string{"A", "B"}), "did not dedup correctly")
|
|
assert.DeepEqual(t, []string{"A", "B"}, dedupEndpoints([]string{"A", "A", "A", "B"}), "did not dedup correctly")
|
|
assert.DeepEqual(t, []string{"A", "B"}, dedupEndpoints([]string{"A", "A", "A", "B", "B"}), "did not dedup correctly")
|
|
}
|
|
|
|
func Test_batchRequestHeaders_UnderflowChecks(t *testing.T) {
|
|
srv := &Service{}
|
|
start := uint64(101)
|
|
end := uint64(100)
|
|
_, err := srv.batchRequestHeaders(start, end)
|
|
require.ErrorContains(t, "cannot be >", err)
|
|
|
|
start = uint64(200)
|
|
end = uint64(100)
|
|
_, err = srv.batchRequestHeaders(start, end)
|
|
require.ErrorContains(t, "cannot be >", err)
|
|
}
|
|
|
|
func TestService_EnsureConsistentPowchainData(t *testing.T) {
|
|
beaconDB := dbutil.SetupDB(t)
|
|
cache, err := depositcache.New()
|
|
require.NoError(t, err)
|
|
|
|
s1, err := NewService(context.Background(), &Web3ServiceConfig{
|
|
BeaconDB: beaconDB,
|
|
DepositCache: cache,
|
|
})
|
|
require.NoError(t, err)
|
|
genState, err := testutil.NewBeaconState()
|
|
require.NoError(t, err)
|
|
assert.NoError(t, genState.SetSlot(1000))
|
|
|
|
require.NoError(t, s1.cfg.BeaconDB.SaveGenesisData(context.Background(), genState))
|
|
require.NoError(t, s1.ensureValidPowchainData(context.Background()))
|
|
|
|
eth1Data, err := s1.cfg.BeaconDB.PowchainData(context.Background())
|
|
assert.NoError(t, err)
|
|
|
|
assert.NotNil(t, eth1Data)
|
|
assert.Equal(t, true, eth1Data.ChainstartData.Chainstarted)
|
|
}
|
|
|
|
func TestService_InitializeCorrectly(t *testing.T) {
|
|
beaconDB := dbutil.SetupDB(t)
|
|
cache, err := depositcache.New()
|
|
require.NoError(t, err)
|
|
|
|
s1, err := NewService(context.Background(), &Web3ServiceConfig{
|
|
BeaconDB: beaconDB,
|
|
DepositCache: cache,
|
|
})
|
|
require.NoError(t, err)
|
|
genState, err := testutil.NewBeaconState()
|
|
require.NoError(t, err)
|
|
assert.NoError(t, genState.SetSlot(1000))
|
|
|
|
require.NoError(t, s1.cfg.BeaconDB.SaveGenesisData(context.Background(), genState))
|
|
require.NoError(t, s1.ensureValidPowchainData(context.Background()))
|
|
|
|
eth1Data, err := s1.cfg.BeaconDB.PowchainData(context.Background())
|
|
assert.NoError(t, err)
|
|
|
|
assert.NoError(t, s1.initializeEth1Data(context.Background(), eth1Data))
|
|
assert.Equal(t, int64(-1), s1.lastReceivedMerkleIndex, "received incorrect last received merkle index")
|
|
}
|
|
|
|
func TestService_EnsureValidPowchainData(t *testing.T) {
|
|
beaconDB := dbutil.SetupDB(t)
|
|
cache, err := depositcache.New()
|
|
require.NoError(t, err)
|
|
|
|
s1, err := NewService(context.Background(), &Web3ServiceConfig{
|
|
BeaconDB: beaconDB,
|
|
DepositCache: cache,
|
|
})
|
|
require.NoError(t, err)
|
|
genState, err := testutil.NewBeaconState()
|
|
require.NoError(t, err)
|
|
assert.NoError(t, genState.SetSlot(1000))
|
|
|
|
require.NoError(t, s1.cfg.BeaconDB.SaveGenesisData(context.Background(), genState))
|
|
|
|
err = s1.cfg.BeaconDB.SavePowchainData(context.Background(), &protodb.ETH1ChainData{
|
|
ChainstartData: &protodb.ChainStartData{Chainstarted: true},
|
|
DepositContainers: []*protodb.DepositContainer{{Index: 1}},
|
|
})
|
|
require.NoError(t, err)
|
|
require.NoError(t, s1.ensureValidPowchainData(context.Background()))
|
|
|
|
eth1Data, err := s1.cfg.BeaconDB.PowchainData(context.Background())
|
|
assert.NoError(t, err)
|
|
|
|
assert.NotNil(t, eth1Data)
|
|
assert.Equal(t, 0, len(eth1Data.DepositContainers))
|
|
}
|
|
|
|
func TestService_ValidateDepositContainers(t *testing.T) {
|
|
beaconDB := dbutil.SetupDB(t)
|
|
cache, err := depositcache.New()
|
|
require.NoError(t, err)
|
|
|
|
s1, err := NewService(context.Background(), &Web3ServiceConfig{
|
|
BeaconDB: beaconDB,
|
|
DepositCache: cache,
|
|
})
|
|
require.NoError(t, err)
|
|
|
|
var tt = []struct {
|
|
name string
|
|
ctrsFunc func() []*protodb.DepositContainer
|
|
expectedRes bool
|
|
}{
|
|
{
|
|
name: "zero containers",
|
|
ctrsFunc: func() []*protodb.DepositContainer {
|
|
return make([]*protodb.DepositContainer, 0)
|
|
},
|
|
expectedRes: true,
|
|
},
|
|
{
|
|
name: "ordered containers",
|
|
ctrsFunc: func() []*protodb.DepositContainer {
|
|
ctrs := make([]*protodb.DepositContainer, 0)
|
|
for i := 0; i < 10; i++ {
|
|
ctrs = append(ctrs, &protodb.DepositContainer{Index: int64(i), Eth1BlockHeight: uint64(i + 10)})
|
|
}
|
|
return ctrs
|
|
},
|
|
expectedRes: true,
|
|
},
|
|
{
|
|
name: "0th container missing",
|
|
ctrsFunc: func() []*protodb.DepositContainer {
|
|
ctrs := make([]*protodb.DepositContainer, 0)
|
|
for i := 1; i < 10; i++ {
|
|
ctrs = append(ctrs, &protodb.DepositContainer{Index: int64(i), Eth1BlockHeight: uint64(i + 10)})
|
|
}
|
|
return ctrs
|
|
},
|
|
expectedRes: false,
|
|
},
|
|
{
|
|
name: "skipped containers",
|
|
ctrsFunc: func() []*protodb.DepositContainer {
|
|
ctrs := make([]*protodb.DepositContainer, 0)
|
|
for i := 0; i < 10; i++ {
|
|
if i == 5 || i == 7 {
|
|
continue
|
|
}
|
|
ctrs = append(ctrs, &protodb.DepositContainer{Index: int64(i), Eth1BlockHeight: uint64(i + 10)})
|
|
}
|
|
return ctrs
|
|
},
|
|
expectedRes: false,
|
|
},
|
|
}
|
|
|
|
for _, test := range tt {
|
|
assert.Equal(t, test.expectedRes, s1.validateDepositContainers(test.ctrsFunc()))
|
|
}
|
|
}
|
|
|
|
func TestTimestampIsChecked(t *testing.T) {
|
|
timestamp := uint64(time.Now().Unix())
|
|
assert.Equal(t, false, eth1HeadIsBehind(timestamp))
|
|
|
|
// Give an older timestmap beyond threshold.
|
|
timestamp = uint64(time.Now().Add(-eth1Threshold).Add(-1 * time.Minute).Unix())
|
|
assert.Equal(t, true, eth1HeadIsBehind(timestamp))
|
|
}
|