bsc/eth/handler_bsc_test.go

260 lines
7.6 KiB
Go
Raw Permalink Normal View History

package eth
import (
"fmt"
"testing"
"time"
"github.com/ethereum/go-ethereum/common"
"github.com/ethereum/go-ethereum/core"
"github.com/ethereum/go-ethereum/core/forkid"
"github.com/ethereum/go-ethereum/core/types"
"github.com/ethereum/go-ethereum/eth/protocols/bsc"
"github.com/ethereum/go-ethereum/eth/protocols/eth"
"github.com/ethereum/go-ethereum/event"
"github.com/ethereum/go-ethereum/p2p"
"github.com/ethereum/go-ethereum/p2p/enode"
)
type testBscHandler struct {
voteBroadcasts event.Feed
}
func (h *testBscHandler) Chain() *core.BlockChain { panic("no backing chain") }
func (h *testBscHandler) RunPeer(peer *bsc.Peer, handler bsc.Handler) error {
panic("not used in tests")
}
func (h *testBscHandler) PeerInfo(enode.ID) interface{} { panic("not used in tests") }
func (h *testBscHandler) Handle(peer *bsc.Peer, packet bsc.Packet) error {
switch packet := packet.(type) {
case *bsc.VotesPacket:
h.voteBroadcasts.Send(packet.Votes)
return nil
default:
panic(fmt.Sprintf("unexpected bsc packet type in tests: %T", packet))
}
}
func TestSendVotes68(t *testing.T) { testSendVotes(t, eth.ETH68) }
func testSendVotes(t *testing.T, protocol uint) {
t.Parallel()
// Create a message handler and fill the pool with big votes
handler := newTestHandler()
defer handler.close()
insert := make([]*types.VoteEnvelope, 100)
for index := range insert {
vote := types.VoteEnvelope{
VoteAddress: types.BLSPublicKey{},
Signature: types.BLSSignature{},
Data: &types.VoteData{
SourceNumber: uint64(0),
SourceHash: common.BytesToHash(common.Hex2Bytes(string(rune(0)))),
TargetNumber: uint64(index),
TargetHash: common.BytesToHash(common.Hex2Bytes(string(rune(index)))),
},
}
insert[index] = &vote
go handler.votepool.PutVote(&vote)
}
time.Sleep(250 * time.Millisecond) // Wait until vote events get out of the system (can't use events, vote broadcaster races with peer join)
protos := []p2p.Protocol{
{
Name: "eth",
Version: eth.ETH68,
},
{
Name: "bsc",
Version: bsc.Bsc1,
},
}
caps := []p2p.Cap{
{
Name: "eth",
Version: eth.ETH68,
},
{
Name: "bsc",
Version: bsc.Bsc1,
},
}
// Create a source handler to send messages through and a sink peer to receive them
p2pEthSrc, p2pEthSink := p2p.MsgPipe()
defer p2pEthSrc.Close()
defer p2pEthSink.Close()
localEth := eth.NewPeer(protocol, p2p.NewPeerWithProtocols(enode.ID{1}, protos, "", caps), p2pEthSrc, nil)
remoteEth := eth.NewPeer(protocol, p2p.NewPeerWithProtocols(enode.ID{2}, protos, "", caps), p2pEthSink, nil)
defer localEth.Close()
defer remoteEth.Close()
p2pBscSrc, p2pBscSink := p2p.MsgPipe()
defer p2pBscSrc.Close()
defer p2pBscSink.Close()
localBsc := bsc.NewPeer(bsc.Bsc1, p2p.NewPeerWithProtocols(enode.ID{1}, protos, "", caps), p2pBscSrc)
remoteBsc := bsc.NewPeer(bsc.Bsc1, p2p.NewPeerWithProtocols(enode.ID{3}, protos, "", caps), p2pBscSink)
defer localBsc.Close()
defer remoteBsc.Close()
go func(p *bsc.Peer) {
(*bscHandler)(handler.handler).RunPeer(p, func(peer *bsc.Peer) error {
return bsc.Handle((*bscHandler)(handler.handler), peer)
})
}(localBsc)
time.Sleep(200 * time.Millisecond)
remoteBsc.Handshake()
time.Sleep(200 * time.Millisecond)
go func(p *eth.Peer) {
handler.handler.runEthPeer(p, func(peer *eth.Peer) error {
return eth.Handle((*ethHandler)(handler.handler), peer)
})
}(localEth)
// Run the handshake locally to avoid spinning up a source handler
var (
genesis = handler.chain.Genesis()
head = handler.chain.CurrentBlock()
Big merge v1.10.16 v1.12.2 fix ci (#1850) * fix: crash of highestVerifiedHeader * fix: panic of blobpool * fix: genesis set up * 1. modify NewDatabaseWithNodeDB to upstream 2. fix race use of hasher in statedb 3. fix use wrong value when updateTrie * fix dir legacypool * fix dir blobpool * fix dir vote * remove diffsync related code * fix core/state/snapshot * disable pipeCommit for now * fix applyTransaction for bloom setting * CI: fast finality in gasprice test * CI: diffFetcher was removed * CI: downloader, remove beaconsync test * CI: no beaconsync in downloader, remove a failed case TestCheckpointChallenge was removed in: https://github.com/ethereum/go-ethereum/pull/27147 since after merge, it is useless for ethereum, but might be useful for BSC. disable the case right now, as it is not a big issue. * CI: bsc protocol decHandlers * CI: receipt Bloom process * 1. skip CheckConfigForkOrder for non-parlia engine 2. all test cases in core work well now cd core && go test ./... -v * fix test cases in trie dir * CI: no beaconsync in downloader, remove a failed case(redo) * fix dir miner * fix dir cmd/geth * CI: filter test, BaseFee & Finality * fix dir graphql * remove diffStore * fix ethclient * fix TestRPCGetTransactionReceipt * fix dir internal * ut add dir ethstats and signer * disable pipeCommit thoroughly; fix concurrent map iteration and map write in statedb * CI: fix snap sync it could be changed by mistake * fix tests/Run to generate snapshot * prepare for merge * remove useless * use common hasher in getDeletedStateObject, no race here * an critical comment for state.Prepare * do not copy nil accessList * add omitempty tag for unused new fields of core.Genesis * remove totalFees * calculate fees before FinalizeAndAssemble * revert interface Finalize of consensus * do not double gas limit upon london block * use Leveldb as default * Revert "remove diffStore" This reverts commit df343b137412b0beb25298a6ba9c3c19e47f20b1. * Revert "remove diffsync related code" This reverts commit 8d84b81feae5d794cb5d7fcfdb7f5f7da751941b. * compile pass after revert * remove diffsync * fix dir eth/protocols/trust * fix TestFastNode * decHandlers for trust protocol * keep persist diff in test
2023-09-07 16:39:29 +08:00
td = handler.chain.GetTd(head.Hash(), head.Number.Uint64())
)
time.Sleep(200 * time.Millisecond)
if err := remoteEth.Handshake(1, td, head.Hash(), genesis.Hash(), forkid.NewIDWithChain(handler.chain), forkid.NewFilter(handler.chain), nil); err != nil {
t.Fatalf("failed to run protocol handshake: %d", err)
}
// After the handshake completes, the source handler should stream the sink
// the votes, subscribe to all inbound network events
backend := new(testBscHandler)
bcasts := make(chan []*types.VoteEnvelope)
bcastSub := backend.voteBroadcasts.Subscribe(bcasts)
defer bcastSub.Unsubscribe()
go bsc.Handle(backend, remoteBsc)
// Make sure we get all the votes on the correct channels
seen := make(map[common.Hash]struct{})
for len(seen) < len(insert) {
votes := <-bcasts
for _, vote := range votes {
if _, ok := seen[vote.Hash()]; ok {
t.Errorf("duplicate vote broadcast: %x", vote.Hash())
}
seen[vote.Hash()] = struct{}{}
}
}
for _, vote := range insert {
if _, ok := seen[vote.Hash()]; !ok {
t.Errorf("missing vote: %x", vote.Hash())
}
}
}
func TestRecvVotes68(t *testing.T) { testRecvVotes(t, eth.ETH68) }
func testRecvVotes(t *testing.T, protocol uint) {
t.Parallel()
// Create a message handler and fill the pool with big votes
handler := newTestHandler()
defer handler.close()
protos := []p2p.Protocol{
{
Name: "eth",
Version: eth.ETH68,
},
{
Name: "bsc",
Version: bsc.Bsc1,
},
}
caps := []p2p.Cap{
{
Name: "eth",
Version: eth.ETH68,
},
{
Name: "bsc",
Version: bsc.Bsc1,
},
}
// Create a source handler to send messages through and a sink peer to receive them
p2pEthSrc, p2pEthSink := p2p.MsgPipe()
defer p2pEthSrc.Close()
defer p2pEthSink.Close()
localEth := eth.NewPeer(protocol, p2p.NewPeerWithProtocols(enode.ID{1}, protos, "", caps), p2pEthSrc, nil)
remoteEth := eth.NewPeer(protocol, p2p.NewPeerWithProtocols(enode.ID{2}, protos, "", caps), p2pEthSink, nil)
defer localEth.Close()
defer remoteEth.Close()
p2pBscSrc, p2pBscSink := p2p.MsgPipe()
defer p2pBscSrc.Close()
defer p2pBscSink.Close()
localBsc := bsc.NewPeer(bsc.Bsc1, p2p.NewPeerWithProtocols(enode.ID{1}, protos, "", caps), p2pBscSrc)
remoteBsc := bsc.NewPeer(bsc.Bsc1, p2p.NewPeerWithProtocols(enode.ID{3}, protos, "", caps), p2pBscSink)
defer localBsc.Close()
defer remoteBsc.Close()
go func(p *bsc.Peer) {
(*bscHandler)(handler.handler).RunPeer(p, func(peer *bsc.Peer) error {
return bsc.Handle((*bscHandler)(handler.handler), peer)
})
}(localBsc)
time.Sleep(200 * time.Millisecond)
remoteBsc.Handshake()
time.Sleep(200 * time.Millisecond)
go func(p *eth.Peer) {
handler.handler.runEthPeer(p, func(peer *eth.Peer) error {
return eth.Handle((*ethHandler)(handler.handler), peer)
})
}(localEth)
// Run the handshake locally to avoid spinning up a source handler
var (
genesis = handler.chain.Genesis()
head = handler.chain.CurrentBlock()
Big merge v1.10.16 v1.12.2 fix ci (#1850) * fix: crash of highestVerifiedHeader * fix: panic of blobpool * fix: genesis set up * 1. modify NewDatabaseWithNodeDB to upstream 2. fix race use of hasher in statedb 3. fix use wrong value when updateTrie * fix dir legacypool * fix dir blobpool * fix dir vote * remove diffsync related code * fix core/state/snapshot * disable pipeCommit for now * fix applyTransaction for bloom setting * CI: fast finality in gasprice test * CI: diffFetcher was removed * CI: downloader, remove beaconsync test * CI: no beaconsync in downloader, remove a failed case TestCheckpointChallenge was removed in: https://github.com/ethereum/go-ethereum/pull/27147 since after merge, it is useless for ethereum, but might be useful for BSC. disable the case right now, as it is not a big issue. * CI: bsc protocol decHandlers * CI: receipt Bloom process * 1. skip CheckConfigForkOrder for non-parlia engine 2. all test cases in core work well now cd core && go test ./... -v * fix test cases in trie dir * CI: no beaconsync in downloader, remove a failed case(redo) * fix dir miner * fix dir cmd/geth * CI: filter test, BaseFee & Finality * fix dir graphql * remove diffStore * fix ethclient * fix TestRPCGetTransactionReceipt * fix dir internal * ut add dir ethstats and signer * disable pipeCommit thoroughly; fix concurrent map iteration and map write in statedb * CI: fix snap sync it could be changed by mistake * fix tests/Run to generate snapshot * prepare for merge * remove useless * use common hasher in getDeletedStateObject, no race here * an critical comment for state.Prepare * do not copy nil accessList * add omitempty tag for unused new fields of core.Genesis * remove totalFees * calculate fees before FinalizeAndAssemble * revert interface Finalize of consensus * do not double gas limit upon london block * use Leveldb as default * Revert "remove diffStore" This reverts commit df343b137412b0beb25298a6ba9c3c19e47f20b1. * Revert "remove diffsync related code" This reverts commit 8d84b81feae5d794cb5d7fcfdb7f5f7da751941b. * compile pass after revert * remove diffsync * fix dir eth/protocols/trust * fix TestFastNode * decHandlers for trust protocol * keep persist diff in test
2023-09-07 16:39:29 +08:00
td = handler.chain.GetTd(head.Hash(), head.Number.Uint64())
)
time.Sleep(200 * time.Millisecond)
if err := remoteEth.Handshake(1, td, head.Hash(), genesis.Hash(), forkid.NewIDWithChain(handler.chain), forkid.NewFilter(handler.chain), nil); err != nil {
t.Fatalf("failed to run protocol handshake: %d", err)
}
votesCh := make(chan core.NewVoteEvent)
sub := handler.votepool.SubscribeNewVoteEvent(votesCh)
defer sub.Unsubscribe()
// Send the vote to the sink and verify that it's added to the vote pool
vote := types.VoteEnvelope{
VoteAddress: types.BLSPublicKey{},
Signature: types.BLSSignature{},
Data: &types.VoteData{
SourceNumber: uint64(0),
SourceHash: common.BytesToHash(common.Hex2Bytes(string(rune(0)))),
TargetNumber: uint64(1),
TargetHash: common.BytesToHash(common.Hex2Bytes(string(rune(1)))),
},
}
remoteBsc.AsyncSendVotes([]*types.VoteEnvelope{&vote})
time.Sleep(100 * time.Millisecond)
select {
case event := <-votesCh:
if event.Vote.Hash() != vote.Hash() {
t.Errorf("added wrong vote hash: got %v, want %v", event.Vote.Hash(), vote.Hash())
}
case <-time.After(2 * time.Second):
t.Errorf("no NewVotesEvent received within 2 seconds")
}
}