USB enumeration still occured. Make sure it will only occur if --usb is set. This also deprecates the 'NoUSB' config file option in favor of a new option 'USB'.
		
			
				
	
	
		
			221 lines
		
	
	
		
			6.5 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			221 lines
		
	
	
		
			6.5 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
package main
 | 
						|
 | 
						|
import (
 | 
						|
	"context"
 | 
						|
	"fmt"
 | 
						|
	"os"
 | 
						|
	"path/filepath"
 | 
						|
	"runtime"
 | 
						|
	"strings"
 | 
						|
	"sync/atomic"
 | 
						|
	"testing"
 | 
						|
	"time"
 | 
						|
 | 
						|
	"github.com/ethereum/go-ethereum/p2p"
 | 
						|
	"github.com/ethereum/go-ethereum/rpc"
 | 
						|
)
 | 
						|
 | 
						|
type gethrpc struct {
 | 
						|
	name     string
 | 
						|
	rpc      *rpc.Client
 | 
						|
	geth     *testgeth
 | 
						|
	nodeInfo *p2p.NodeInfo
 | 
						|
}
 | 
						|
 | 
						|
func (g *gethrpc) killAndWait() {
 | 
						|
	g.geth.Kill()
 | 
						|
	g.geth.WaitExit()
 | 
						|
}
 | 
						|
 | 
						|
func (g *gethrpc) callRPC(result interface{}, method string, args ...interface{}) {
 | 
						|
	if err := g.rpc.Call(&result, method, args...); err != nil {
 | 
						|
		g.geth.Fatalf("callRPC %v: %v", method, err)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func (g *gethrpc) addPeer(peer *gethrpc) {
 | 
						|
	g.geth.Logf("%v.addPeer(%v)", g.name, peer.name)
 | 
						|
	enode := peer.getNodeInfo().Enode
 | 
						|
	peerCh := make(chan *p2p.PeerEvent)
 | 
						|
	sub, err := g.rpc.Subscribe(context.Background(), "admin", peerCh, "peerEvents")
 | 
						|
	if err != nil {
 | 
						|
		g.geth.Fatalf("subscribe %v: %v", g.name, err)
 | 
						|
	}
 | 
						|
	defer sub.Unsubscribe()
 | 
						|
	g.callRPC(nil, "admin_addPeer", enode)
 | 
						|
	dur := 14 * time.Second
 | 
						|
	timeout := time.After(dur)
 | 
						|
	select {
 | 
						|
	case ev := <-peerCh:
 | 
						|
		g.geth.Logf("%v received event: type=%v, peer=%v", g.name, ev.Type, ev.Peer)
 | 
						|
	case err := <-sub.Err():
 | 
						|
		g.geth.Fatalf("%v sub error: %v", g.name, err)
 | 
						|
	case <-timeout:
 | 
						|
		g.geth.Error("timeout adding peer after", dur)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
// Use this function instead of `g.nodeInfo` directly
 | 
						|
func (g *gethrpc) getNodeInfo() *p2p.NodeInfo {
 | 
						|
	if g.nodeInfo != nil {
 | 
						|
		return g.nodeInfo
 | 
						|
	}
 | 
						|
	g.nodeInfo = &p2p.NodeInfo{}
 | 
						|
	g.callRPC(&g.nodeInfo, "admin_nodeInfo")
 | 
						|
	return g.nodeInfo
 | 
						|
}
 | 
						|
 | 
						|
func (g *gethrpc) waitSynced() {
 | 
						|
	// Check if it's synced now
 | 
						|
	var result interface{}
 | 
						|
	g.callRPC(&result, "eth_syncing")
 | 
						|
	syncing, ok := result.(bool)
 | 
						|
	if ok && !syncing {
 | 
						|
		g.geth.Logf("%v already synced", g.name)
 | 
						|
		return
 | 
						|
	}
 | 
						|
 | 
						|
	// Actually wait, subscribe to the event
 | 
						|
	ch := make(chan interface{})
 | 
						|
	sub, err := g.rpc.Subscribe(context.Background(), "eth", ch, "syncing")
 | 
						|
	if err != nil {
 | 
						|
		g.geth.Fatalf("%v syncing: %v", g.name, err)
 | 
						|
	}
 | 
						|
	defer sub.Unsubscribe()
 | 
						|
	timeout := time.After(4 * time.Second)
 | 
						|
	select {
 | 
						|
	case ev := <-ch:
 | 
						|
		g.geth.Log("'syncing' event", ev)
 | 
						|
		syncing, ok := ev.(bool)
 | 
						|
		if ok && !syncing {
 | 
						|
			break
 | 
						|
		}
 | 
						|
		g.geth.Log("Other 'syncing' event", ev)
 | 
						|
	case err := <-sub.Err():
 | 
						|
		g.geth.Fatalf("%v notification: %v", g.name, err)
 | 
						|
		break
 | 
						|
	case <-timeout:
 | 
						|
		g.geth.Fatalf("%v timeout syncing", g.name)
 | 
						|
		break
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
// ipcEndpoint resolves an IPC endpoint based on a configured value, taking into
 | 
						|
// account the set data folders as well as the designated platform we're currently
 | 
						|
// running on.
 | 
						|
func ipcEndpoint(ipcPath, datadir string) string {
 | 
						|
	// On windows we can only use plain top-level pipes
 | 
						|
	if runtime.GOOS == "windows" {
 | 
						|
		if strings.HasPrefix(ipcPath, `\\.\pipe\`) {
 | 
						|
			return ipcPath
 | 
						|
		}
 | 
						|
		return `\\.\pipe\` + ipcPath
 | 
						|
	}
 | 
						|
	// Resolve names into the data directory full paths otherwise
 | 
						|
	if filepath.Base(ipcPath) == ipcPath {
 | 
						|
		if datadir == "" {
 | 
						|
			return filepath.Join(os.TempDir(), ipcPath)
 | 
						|
		}
 | 
						|
		return filepath.Join(datadir, ipcPath)
 | 
						|
	}
 | 
						|
	return ipcPath
 | 
						|
}
 | 
						|
 | 
						|
// nextIPC ensures that each ipc pipe gets a unique name.
 | 
						|
// On linux, it works well to use ipc pipes all over the filesystem (in datadirs),
 | 
						|
// but windows require pipes to sit in "\\.\pipe\". Therefore, to run several
 | 
						|
// nodes simultaneously, we need to distinguish between them, which we do by
 | 
						|
// the pipe filename instead of folder.
 | 
						|
var nextIPC = uint32(0)
 | 
						|
 | 
						|
func startGethWithIpc(t *testing.T, name string, args ...string) *gethrpc {
 | 
						|
	ipcName := fmt.Sprintf("geth-%d.ipc", atomic.AddUint32(&nextIPC, 1))
 | 
						|
	args = append([]string{"--networkid=42", "--port=0", "--ipcpath", ipcName}, args...)
 | 
						|
	t.Logf("Starting %v with rpc: %v", name, args)
 | 
						|
 | 
						|
	g := &gethrpc{
 | 
						|
		name: name,
 | 
						|
		geth: runGeth(t, args...),
 | 
						|
	}
 | 
						|
	// wait before we can attach to it. TODO: probe for it properly
 | 
						|
	time.Sleep(1 * time.Second)
 | 
						|
	var err error
 | 
						|
	ipcpath := ipcEndpoint(ipcName, g.geth.Datadir)
 | 
						|
	if g.rpc, err = rpc.Dial(ipcpath); err != nil {
 | 
						|
		t.Fatalf("%v rpc connect to %v: %v", name, ipcpath, err)
 | 
						|
	}
 | 
						|
	return g
 | 
						|
}
 | 
						|
 | 
						|
func initGeth(t *testing.T) string {
 | 
						|
	args := []string{"--networkid=42", "init", "./testdata/clique.json"}
 | 
						|
	t.Logf("Initializing geth: %v ", args)
 | 
						|
	g := runGeth(t, args...)
 | 
						|
	datadir := g.Datadir
 | 
						|
	g.WaitExit()
 | 
						|
	return datadir
 | 
						|
}
 | 
						|
 | 
						|
func startLightServer(t *testing.T) *gethrpc {
 | 
						|
	datadir := initGeth(t)
 | 
						|
	t.Logf("Importing keys to geth")
 | 
						|
	runGeth(t, "--datadir", datadir, "--password", "./testdata/password.txt", "account", "import", "./testdata/key.prv", "--lightkdf").WaitExit()
 | 
						|
	account := "0x02f0d131f1f97aef08aec6e3291b957d9efe7105"
 | 
						|
	server := startGethWithIpc(t, "lightserver", "--allow-insecure-unlock", "--datadir", datadir, "--password", "./testdata/password.txt", "--unlock", account, "--mine", "--light.serve=100", "--light.maxpeers=1", "--nodiscover", "--nat=extip:127.0.0.1", "--verbosity=4")
 | 
						|
	return server
 | 
						|
}
 | 
						|
 | 
						|
func startClient(t *testing.T, name string) *gethrpc {
 | 
						|
	datadir := initGeth(t)
 | 
						|
	return startGethWithIpc(t, name, "--datadir", datadir, "--nodiscover", "--syncmode=light", "--nat=extip:127.0.0.1", "--verbosity=4")
 | 
						|
}
 | 
						|
 | 
						|
func TestPriorityClient(t *testing.T) {
 | 
						|
	lightServer := startLightServer(t)
 | 
						|
	defer lightServer.killAndWait()
 | 
						|
 | 
						|
	// Start client and add lightServer as peer
 | 
						|
	freeCli := startClient(t, "freeCli")
 | 
						|
	defer freeCli.killAndWait()
 | 
						|
	freeCli.addPeer(lightServer)
 | 
						|
 | 
						|
	var peers []*p2p.PeerInfo
 | 
						|
	freeCli.callRPC(&peers, "admin_peers")
 | 
						|
	if len(peers) != 1 {
 | 
						|
		t.Errorf("Expected: # of client peers == 1, actual: %v", len(peers))
 | 
						|
		return
 | 
						|
	}
 | 
						|
 | 
						|
	// Set up priority client, get its nodeID, increase its balance on the lightServer
 | 
						|
	prioCli := startClient(t, "prioCli")
 | 
						|
	defer prioCli.killAndWait()
 | 
						|
	// 3_000_000_000 once we move to Go 1.13
 | 
						|
	tokens := uint64(3000000000)
 | 
						|
	lightServer.callRPC(nil, "les_addBalance", prioCli.getNodeInfo().ID, tokens)
 | 
						|
	prioCli.addPeer(lightServer)
 | 
						|
 | 
						|
	// Check if priority client is actually syncing and the regular client got kicked out
 | 
						|
	prioCli.callRPC(&peers, "admin_peers")
 | 
						|
	if len(peers) != 1 {
 | 
						|
		t.Errorf("Expected: # of prio peers == 1, actual: %v", len(peers))
 | 
						|
	}
 | 
						|
 | 
						|
	nodes := map[string]*gethrpc{
 | 
						|
		lightServer.getNodeInfo().ID: lightServer,
 | 
						|
		freeCli.getNodeInfo().ID:     freeCli,
 | 
						|
		prioCli.getNodeInfo().ID:     prioCli,
 | 
						|
	}
 | 
						|
	time.Sleep(1 * time.Second)
 | 
						|
	lightServer.callRPC(&peers, "admin_peers")
 | 
						|
	peersWithNames := make(map[string]string)
 | 
						|
	for _, p := range peers {
 | 
						|
		peersWithNames[nodes[p.ID].name] = p.ID
 | 
						|
	}
 | 
						|
	if _, freeClientFound := peersWithNames[freeCli.name]; freeClientFound {
 | 
						|
		t.Error("client is still a peer of lightServer", peersWithNames)
 | 
						|
	}
 | 
						|
	if _, prioClientFound := peersWithNames[prioCli.name]; !prioClientFound {
 | 
						|
		t.Error("prio client is not among lightServer peers", peersWithNames)
 | 
						|
	}
 | 
						|
}
 |