mirror of
https://source.quilibrium.com/quilibrium/ceremonyclient.git
synced 2025-01-15 10:15:17 +00:00
278 lines
6.0 KiB
Go
278 lines
6.0 KiB
Go
package reuseport
|
|
|
|
import (
|
|
"context"
|
|
"net"
|
|
"runtime"
|
|
"testing"
|
|
"time"
|
|
|
|
ma "github.com/multiformats/go-multiaddr"
|
|
manet "github.com/multiformats/go-multiaddr/net"
|
|
)
|
|
|
|
var loopbackV4, _ = ma.NewMultiaddr("/ip4/127.0.0.1/tcp/0")
|
|
var loopbackV6, _ = ma.NewMultiaddr("/ip6/::1/tcp/0")
|
|
var unspecV6, _ = ma.NewMultiaddr("/ip6/::/tcp/0")
|
|
var unspecV4, _ = ma.NewMultiaddr("/ip4/0.0.0.0/tcp/0")
|
|
|
|
var globalV4 ma.Multiaddr
|
|
var globalV6 ma.Multiaddr
|
|
|
|
func init() {
|
|
addrs, err := manet.InterfaceMultiaddrs()
|
|
if err != nil {
|
|
return
|
|
}
|
|
for _, addr := range addrs {
|
|
if !manet.IsIP6LinkLocal(addr) && !manet.IsIPLoopback(addr) {
|
|
tcp, _ := ma.NewMultiaddr("/tcp/0")
|
|
switch addr.Protocols()[0].Code {
|
|
case ma.P_IP4:
|
|
if globalV4 == nil {
|
|
globalV4 = addr.Encapsulate(tcp)
|
|
}
|
|
case ma.P_IP6:
|
|
if globalV6 == nil {
|
|
globalV6 = addr.Encapsulate(tcp)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func setLingerZero(c manet.Conn) {
|
|
if runtime.GOOS == "darwin" {
|
|
c.(interface{ SetLinger(int) error }).SetLinger(0)
|
|
}
|
|
}
|
|
|
|
func acceptOne(t *testing.T, listener manet.Listener) <-chan manet.Conn {
|
|
t.Helper()
|
|
done := make(chan manet.Conn, 1)
|
|
go func() {
|
|
defer close(done)
|
|
c, err := listener.Accept()
|
|
if err != nil {
|
|
t.Error(err)
|
|
return
|
|
}
|
|
setLingerZero(c)
|
|
done <- c
|
|
}()
|
|
return done
|
|
}
|
|
|
|
func dialOne(t *testing.T, tr *Transport, listener manet.Listener, expected ...int) int {
|
|
t.Helper()
|
|
|
|
ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second)
|
|
defer cancel()
|
|
|
|
connChan := acceptOne(t, listener)
|
|
c, err := tr.DialContext(ctx, listener.Multiaddr())
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
setLingerZero(c)
|
|
port := c.LocalAddr().(*net.TCPAddr).Port
|
|
serverConn := <-connChan
|
|
serverConn.Close()
|
|
c.Close()
|
|
if len(expected) == 0 {
|
|
return port
|
|
}
|
|
for _, p := range expected {
|
|
if p == port {
|
|
return port
|
|
}
|
|
}
|
|
t.Errorf("dialed %s from %v. expected to dial from port %v", listener.Multiaddr(), c.LocalAddr(), expected)
|
|
return 0
|
|
}
|
|
|
|
func TestNoneAndSingle(t *testing.T) {
|
|
var trA Transport
|
|
var trB Transport
|
|
listenerA, err := trA.Listen(loopbackV4)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer listenerA.Close()
|
|
|
|
dialOne(t, &trB, listenerA)
|
|
|
|
listenerB, err := trB.Listen(loopbackV4)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer listenerB.Close()
|
|
|
|
dialOne(t, &trB, listenerA, listenerB.Addr().(*net.TCPAddr).Port)
|
|
}
|
|
|
|
func TestTwoLocal(t *testing.T) {
|
|
var trA Transport
|
|
var trB Transport
|
|
listenerA, err := trA.Listen(loopbackV4)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer listenerA.Close()
|
|
|
|
listenerB1, err := trB.Listen(loopbackV4)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer listenerB1.Close()
|
|
|
|
listenerB2, err := trB.Listen(loopbackV4)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer listenerB2.Close()
|
|
|
|
dialOne(t, &trB, listenerA,
|
|
listenerB1.Addr().(*net.TCPAddr).Port,
|
|
listenerB2.Addr().(*net.TCPAddr).Port)
|
|
}
|
|
|
|
func TestGlobalPreferenceV4(t *testing.T) {
|
|
if globalV4 == nil {
|
|
t.Skip("no global IPv4 addresses configured")
|
|
return
|
|
}
|
|
t.Logf("when listening on %v, should prefer %v over %v", loopbackV4, loopbackV4, globalV4)
|
|
testPrefer(t, loopbackV4, loopbackV4, globalV4)
|
|
t.Logf("when listening on %v, should prefer %v over %v", loopbackV4, unspecV4, globalV4)
|
|
testPrefer(t, loopbackV4, unspecV4, globalV4)
|
|
t.Logf("when listening on %v, should prefer %v over %v", globalV4, unspecV4, loopbackV4)
|
|
testPrefer(t, globalV4, unspecV4, loopbackV4)
|
|
}
|
|
|
|
func TestGlobalPreferenceV6(t *testing.T) {
|
|
if globalV6 == nil {
|
|
t.Skip("no global IPv6 addresses configured")
|
|
return
|
|
}
|
|
testPrefer(t, loopbackV6, loopbackV6, globalV6)
|
|
testPrefer(t, loopbackV6, unspecV6, globalV6)
|
|
|
|
testPrefer(t, globalV6, unspecV6, loopbackV6)
|
|
}
|
|
|
|
func TestLoopbackPreference(t *testing.T) {
|
|
testPrefer(t, loopbackV4, loopbackV4, unspecV4)
|
|
testPrefer(t, loopbackV6, loopbackV6, unspecV6)
|
|
}
|
|
|
|
func testPrefer(t *testing.T, listen, prefer, avoid ma.Multiaddr) {
|
|
var trA Transport
|
|
var trB Transport
|
|
listenerA, err := trA.Listen(listen)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer listenerA.Close()
|
|
|
|
listenerB1, err := trB.Listen(avoid)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer listenerB1.Close()
|
|
|
|
listenerB2, err := trB.Listen(prefer)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer listenerB2.Close()
|
|
|
|
dialOne(t, &trB, listenerA, listenerB2.Addr().(*net.TCPAddr).Port)
|
|
}
|
|
|
|
func TestV6V4(t *testing.T) {
|
|
if runtime.GOOS == "darwin" {
|
|
t.Skip("This test is failing on OSX: https://github.com/libp2p/go-reuseport-transport/issues/40")
|
|
}
|
|
testUseFirst(t, loopbackV4, loopbackV4, loopbackV6)
|
|
testUseFirst(t, loopbackV6, loopbackV6, loopbackV4)
|
|
}
|
|
|
|
func TestGlobalToGlobal(t *testing.T) {
|
|
if globalV4 == nil {
|
|
t.Skip("no globalV4 addresses configured")
|
|
return
|
|
}
|
|
testUseFirst(t, globalV4, globalV4, loopbackV4)
|
|
testUseFirst(t, globalV6, globalV6, loopbackV6)
|
|
}
|
|
|
|
func testUseFirst(t *testing.T, listen, use, never ma.Multiaddr) {
|
|
var trA Transport
|
|
var trB Transport
|
|
listenerA, err := trA.Listen(globalV4)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer listenerA.Close()
|
|
|
|
listenerB1, err := trB.Listen(loopbackV4)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer listenerB1.Close()
|
|
|
|
// It works (random port)
|
|
dialOne(t, &trB, listenerA)
|
|
|
|
listenerB2, err := trB.Listen(globalV4)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer listenerB2.Close()
|
|
|
|
// Uses globalV4 port.
|
|
dialOne(t, &trB, listenerA, listenerB2.Addr().(*net.TCPAddr).Port)
|
|
|
|
// Closing the listener should reset the dialer.
|
|
listenerB2.Close()
|
|
|
|
// It still works.
|
|
dialOne(t, &trB, listenerA)
|
|
}
|
|
|
|
func TestDuplicateGlobal(t *testing.T) {
|
|
if globalV4 == nil {
|
|
t.Skip("no globalV4 addresses configured")
|
|
return
|
|
}
|
|
|
|
var trA Transport
|
|
var trB Transport
|
|
listenerA, err := trA.Listen(globalV4)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer listenerA.Close()
|
|
|
|
listenerB1, err := trB.Listen(globalV4)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer listenerB1.Close()
|
|
|
|
listenerB2, err := trB.Listen(globalV4)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer listenerB2.Close()
|
|
|
|
// Check which port we're using
|
|
port := dialOne(t, &trB, listenerA)
|
|
|
|
// Check consistency
|
|
for i := 0; i < 10; i++ {
|
|
dialOne(t, &trB, listenerA, port)
|
|
}
|
|
}
|