2020-07-01 11:31:11 +03:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2020-11-20 00:50:47 +03:00
|
|
|
"fmt"
|
2020-11-17 14:01:19 +03:00
|
|
|
"os"
|
2020-07-01 11:31:11 +03:00
|
|
|
"path/filepath"
|
2020-11-17 14:01:19 +03:00
|
|
|
"runtime"
|
|
|
|
"strings"
|
2020-11-20 00:50:47 +03:00
|
|
|
"sync/atomic"
|
2020-07-01 11:31:11 +03:00
|
|
|
"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
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-17 14:01:19 +03:00
|
|
|
// 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
|
|
|
|
}
|
|
|
|
|
2020-11-20 00:50:47 +03:00
|
|
|
// 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)
|
|
|
|
|
2020-08-14 15:18:12 +03:00
|
|
|
func startGethWithIpc(t *testing.T, name string, args ...string) *gethrpc {
|
2020-11-20 00:50:47 +03:00
|
|
|
ipcName := fmt.Sprintf("geth-%d.ipc", atomic.AddUint32(&nextIPC, 1))
|
2021-01-13 13:14:36 +03:00
|
|
|
args = append([]string{"--networkid=42", "--port=0", "--ipcpath", ipcName}, args...)
|
2020-07-01 11:31:11 +03:00
|
|
|
t.Logf("Starting %v with rpc: %v", name, args)
|
2020-11-20 00:50:47 +03:00
|
|
|
|
|
|
|
g := &gethrpc{
|
|
|
|
name: name,
|
|
|
|
geth: runGeth(t, args...),
|
|
|
|
}
|
2020-07-01 11:31:11 +03:00
|
|
|
// wait before we can attach to it. TODO: probe for it properly
|
|
|
|
time.Sleep(1 * time.Second)
|
|
|
|
var err error
|
2020-11-20 00:50:47 +03:00
|
|
|
ipcpath := ipcEndpoint(ipcName, g.geth.Datadir)
|
|
|
|
if g.rpc, err = rpc.Dial(ipcpath); err != nil {
|
2020-11-17 14:01:19 +03:00
|
|
|
t.Fatalf("%v rpc connect to %v: %v", name, ipcpath, err)
|
2020-07-01 11:31:11 +03:00
|
|
|
}
|
|
|
|
return g
|
|
|
|
}
|
|
|
|
|
|
|
|
func initGeth(t *testing.T) string {
|
2021-01-13 13:14:36 +03:00
|
|
|
args := []string{"--networkid=42", "init", "./testdata/clique.json"}
|
2020-11-20 00:50:47 +03:00
|
|
|
t.Logf("Initializing geth: %v ", args)
|
|
|
|
g := runGeth(t, args...)
|
2020-07-01 11:31:11 +03:00
|
|
|
datadir := g.Datadir
|
|
|
|
g.WaitExit()
|
|
|
|
return datadir
|
|
|
|
}
|
|
|
|
|
|
|
|
func startLightServer(t *testing.T) *gethrpc {
|
|
|
|
datadir := initGeth(t)
|
2020-11-20 00:50:47 +03:00
|
|
|
t.Logf("Importing keys to geth")
|
2021-01-13 13:14:36 +03:00
|
|
|
runGeth(t, "--datadir", datadir, "--password", "./testdata/password.txt", "account", "import", "./testdata/key.prv", "--lightkdf").WaitExit()
|
2020-07-01 11:31:11 +03:00
|
|
|
account := "0x02f0d131f1f97aef08aec6e3291b957d9efe7105"
|
2020-11-20 00:50:47 +03:00
|
|
|
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")
|
2020-07-01 11:31:11 +03:00
|
|
|
return server
|
|
|
|
}
|
|
|
|
|
|
|
|
func startClient(t *testing.T, name string) *gethrpc {
|
|
|
|
datadir := initGeth(t)
|
2020-11-20 00:50:47 +03:00
|
|
|
return startGethWithIpc(t, name, "--datadir", datadir, "--nodiscover", "--syncmode=light", "--nat=extip:127.0.0.1", "--verbosity=4")
|
2020-07-01 11:31:11 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
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
|
2020-11-20 00:50:47 +03:00
|
|
|
tokens := uint64(3000000000)
|
2020-09-14 23:44:20 +03:00
|
|
|
lightServer.callRPC(nil, "les_addBalance", prioCli.getNodeInfo().ID, tokens)
|
2020-07-01 11:31:11 +03:00
|
|
|
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,
|
|
|
|
}
|
2020-08-14 15:18:12 +03:00
|
|
|
time.Sleep(1 * time.Second)
|
2020-07-01 11:31:11 +03:00
|
|
|
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)
|
|
|
|
}
|
|
|
|
}
|