github.com/letterj/go-ethereum@v1.8.22-0.20190204142846-520024dfd689/les/ulc_test.go (about)

     1  package les
     2  
     3  import (
     4  	"fmt"
     5  	"reflect"
     6  	"testing"
     7  	"time"
     8  
     9  	"net"
    10  
    11  	"crypto/ecdsa"
    12  	"math/big"
    13  
    14  	"github.com/ethereum/go-ethereum/core"
    15  	"github.com/ethereum/go-ethereum/crypto"
    16  	"github.com/ethereum/go-ethereum/eth"
    17  	"github.com/ethereum/go-ethereum/ethdb"
    18  	"github.com/ethereum/go-ethereum/light"
    19  	"github.com/ethereum/go-ethereum/p2p"
    20  	"github.com/ethereum/go-ethereum/p2p/enode"
    21  )
    22  
    23  func TestULCSyncWithOnePeer(t *testing.T) {
    24  	f := newFullPeerPair(t, 1, 4, testChainGen)
    25  	ulcConfig := &eth.ULCConfig{
    26  		MinTrustedFraction: 100,
    27  		TrustedServers:     []string{f.Node.String()},
    28  	}
    29  
    30  	l := newLightPeer(t, ulcConfig)
    31  
    32  	if reflect.DeepEqual(f.PM.blockchain.CurrentHeader().Hash(), l.PM.blockchain.CurrentHeader().Hash()) {
    33  		t.Fatal("blocks are equal")
    34  	}
    35  
    36  	_, _, err := connectPeers(f, l, 2)
    37  	if err != nil {
    38  		t.Fatal(err)
    39  	}
    40  
    41  	l.PM.fetcher.lock.Lock()
    42  	l.PM.fetcher.nextRequest()
    43  	l.PM.fetcher.lock.Unlock()
    44  
    45  	if !reflect.DeepEqual(f.PM.blockchain.CurrentHeader().Hash(), l.PM.blockchain.CurrentHeader().Hash()) {
    46  		t.Fatal("sync doesn't work")
    47  	}
    48  }
    49  
    50  func TestULCReceiveAnnounce(t *testing.T) {
    51  	f := newFullPeerPair(t, 1, 4, testChainGen)
    52  	ulcConfig := &eth.ULCConfig{
    53  		MinTrustedFraction: 100,
    54  		TrustedServers:     []string{f.Node.String()},
    55  	}
    56  
    57  	l := newLightPeer(t, ulcConfig)
    58  	fPeer, lPeer, err := connectPeers(f, l, 2)
    59  	if err != nil {
    60  		t.Fatal(err)
    61  	}
    62  
    63  	l.PM.synchronise(fPeer)
    64  
    65  	//check that the sync is finished correctly
    66  	if !reflect.DeepEqual(f.PM.blockchain.CurrentHeader().Hash(), l.PM.blockchain.CurrentHeader().Hash()) {
    67  		t.Fatal("sync doesn't work")
    68  	}
    69  
    70  	l.PM.peers.lock.Lock()
    71  	if len(l.PM.peers.peers) == 0 {
    72  		t.Fatal("peer list should not be empty")
    73  	}
    74  	l.PM.peers.lock.Unlock()
    75  
    76  	time.Sleep(time.Second)
    77  	//send a signed announce message(payload doesn't matter)
    78  	td := f.PM.blockchain.GetTd(l.PM.blockchain.CurrentHeader().Hash(), l.PM.blockchain.CurrentHeader().Number.Uint64())
    79  	announce := announceData{
    80  		Number: l.PM.blockchain.CurrentHeader().Number.Uint64() + 1,
    81  		Td:     td.Add(td, big.NewInt(1)),
    82  	}
    83  	announce.sign(f.Key)
    84  	lPeer.SendAnnounce(announce)
    85  }
    86  
    87  func TestULCShouldNotSyncWithTwoPeersOneHaveEmptyChain(t *testing.T) {
    88  	f1 := newFullPeerPair(t, 1, 4, testChainGen)
    89  	f2 := newFullPeerPair(t, 2, 0, nil)
    90  	ulcConf := &ulc{minTrustedFraction: 100, trustedKeys: make(map[string]struct{})}
    91  	ulcConf.trustedKeys[f1.Node.ID().String()] = struct{}{}
    92  	ulcConf.trustedKeys[f2.Node.ID().String()] = struct{}{}
    93  	ulcConfig := &eth.ULCConfig{
    94  		MinTrustedFraction: 100,
    95  		TrustedServers:     []string{f1.Node.String(), f2.Node.String()},
    96  	}
    97  	l := newLightPeer(t, ulcConfig)
    98  	l.PM.ulc.minTrustedFraction = 100
    99  
   100  	_, _, err := connectPeers(f1, l, 2)
   101  	if err != nil {
   102  		t.Fatal(err)
   103  	}
   104  	_, _, err = connectPeers(f2, l, 2)
   105  	if err != nil {
   106  		t.Fatal(err)
   107  	}
   108  
   109  	l.PM.fetcher.lock.Lock()
   110  	l.PM.fetcher.nextRequest()
   111  	l.PM.fetcher.lock.Unlock()
   112  
   113  	if reflect.DeepEqual(f2.PM.blockchain.CurrentHeader().Hash(), l.PM.blockchain.CurrentHeader().Hash()) {
   114  		t.Fatal("Incorrect hash: second peer has empty chain")
   115  	}
   116  }
   117  
   118  func TestULCShouldNotSyncWithThreePeersOneHaveEmptyChain(t *testing.T) {
   119  	f1 := newFullPeerPair(t, 1, 3, testChainGen)
   120  	f2 := newFullPeerPair(t, 2, 4, testChainGen)
   121  	f3 := newFullPeerPair(t, 3, 0, nil)
   122  
   123  	ulcConfig := &eth.ULCConfig{
   124  		MinTrustedFraction: 60,
   125  		TrustedServers:     []string{f1.Node.String(), f2.Node.String(), f3.Node.String()},
   126  	}
   127  
   128  	l := newLightPeer(t, ulcConfig)
   129  	_, _, err := connectPeers(f1, l, 2)
   130  	if err != nil {
   131  		t.Fatal(err)
   132  	}
   133  
   134  	_, _, err = connectPeers(f2, l, 2)
   135  	if err != nil {
   136  		t.Fatal(err)
   137  	}
   138  
   139  	_, _, err = connectPeers(f3, l, 2)
   140  	if err != nil {
   141  		t.Fatal(err)
   142  	}
   143  
   144  	l.PM.fetcher.lock.Lock()
   145  	l.PM.fetcher.nextRequest()
   146  	l.PM.fetcher.lock.Unlock()
   147  
   148  	if !reflect.DeepEqual(f1.PM.blockchain.CurrentHeader().Hash(), l.PM.blockchain.CurrentHeader().Hash()) {
   149  		t.Fatal("Incorrect hash")
   150  	}
   151  }
   152  
   153  type pairPeer struct {
   154  	Name string
   155  	Node *enode.Node
   156  	PM   *ProtocolManager
   157  	Key  *ecdsa.PrivateKey
   158  }
   159  
   160  func connectPeers(full, light pairPeer, version int) (*peer, *peer, error) {
   161  	// Create a message pipe to communicate through
   162  	app, net := p2p.MsgPipe()
   163  
   164  	peerLight := full.PM.newPeer(version, NetworkId, p2p.NewPeer(light.Node.ID(), light.Name, nil), net)
   165  	peerFull := light.PM.newPeer(version, NetworkId, p2p.NewPeer(full.Node.ID(), full.Name, nil), app)
   166  
   167  	// Start the peerLight on a new thread
   168  	errc1 := make(chan error, 1)
   169  	errc2 := make(chan error, 1)
   170  	go func() {
   171  		select {
   172  		case light.PM.newPeerCh <- peerFull:
   173  			errc1 <- light.PM.handle(peerFull)
   174  		case <-light.PM.quitSync:
   175  			errc1 <- p2p.DiscQuitting
   176  		}
   177  	}()
   178  	go func() {
   179  		select {
   180  		case full.PM.newPeerCh <- peerLight:
   181  			errc2 <- full.PM.handle(peerLight)
   182  		case <-full.PM.quitSync:
   183  			errc2 <- p2p.DiscQuitting
   184  		}
   185  	}()
   186  
   187  	select {
   188  	case <-time.After(time.Millisecond * 100):
   189  	case err := <-errc1:
   190  		return nil, nil, fmt.Errorf("peerLight handshake error: %v", err)
   191  	case err := <-errc2:
   192  		return nil, nil, fmt.Errorf("peerFull handshake error: %v", err)
   193  	}
   194  
   195  	return peerFull, peerLight, nil
   196  }
   197  
   198  // newFullPeerPair creates node with full sync mode
   199  func newFullPeerPair(t *testing.T, index int, numberOfblocks int, chainGen func(int, *core.BlockGen)) pairPeer {
   200  	db := ethdb.NewMemDatabase()
   201  
   202  	pmFull := newTestProtocolManagerMust(t, false, numberOfblocks, chainGen, nil, nil, db, nil)
   203  
   204  	peerPairFull := pairPeer{
   205  		Name: "full node",
   206  		PM:   pmFull,
   207  	}
   208  	key, err := crypto.GenerateKey()
   209  	if err != nil {
   210  		t.Fatal("generate key err:", err)
   211  	}
   212  	peerPairFull.Key = key
   213  	peerPairFull.Node = enode.NewV4(&key.PublicKey, net.ParseIP("127.0.0.1"), 35000, 35000)
   214  	return peerPairFull
   215  }
   216  
   217  // newLightPeer creates node with light sync mode
   218  func newLightPeer(t *testing.T, ulcConfig *eth.ULCConfig) pairPeer {
   219  	peers := newPeerSet()
   220  	dist := newRequestDistributor(peers, make(chan struct{}))
   221  	rm := newRetrieveManager(peers, dist, nil)
   222  	ldb := ethdb.NewMemDatabase()
   223  
   224  	odr := NewLesOdr(ldb, light.DefaultClientIndexerConfig, rm)
   225  
   226  	pmLight := newTestProtocolManagerMust(t, true, 0, nil, odr, peers, ldb, ulcConfig)
   227  	peerPairLight := pairPeer{
   228  		Name: "ulc node",
   229  		PM:   pmLight,
   230  	}
   231  
   232  	key, err := crypto.GenerateKey()
   233  	if err != nil {
   234  		t.Fatal("generate key err:", err)
   235  	}
   236  	peerPairLight.Key = key
   237  	peerPairLight.Node = enode.NewV4(&key.PublicKey, net.IP{}, 35000, 35000)
   238  	return peerPairLight
   239  }