2020-04-01 04:14:43 +08:00
|
|
|
package torrent
|
|
|
|
|
|
|
|
import (
|
|
|
|
"net"
|
|
|
|
"testing"
|
|
|
|
|
2020-10-23 05:58:55 +08:00
|
|
|
"github.com/stretchr/testify/assert"
|
2020-04-01 04:14:43 +08:00
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
|
|
|
|
"github.com/anacrolix/dht/v2/krpc"
|
|
|
|
pp "github.com/anacrolix/torrent/peer_protocol"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
addrs = []net.Addr{
|
|
|
|
&net.TCPAddr{IP: net.IPv6loopback, Port: 4747},
|
|
|
|
&net.TCPAddr{IP: net.IPv6loopback, Port: 4748},
|
|
|
|
&net.TCPAddr{IP: net.IPv4(127, 0, 0, 1), Port: 4747},
|
|
|
|
&net.TCPAddr{IP: net.IPv4(127, 0, 0, 1), Port: 4748},
|
|
|
|
}
|
|
|
|
f = pp.PexOutgoingConn
|
|
|
|
)
|
|
|
|
|
|
|
|
func TestPexAdded(t *testing.T) {
|
|
|
|
t.Run("noHold", func(t *testing.T) {
|
|
|
|
s := new(pexState)
|
2020-07-15 14:15:38 +08:00
|
|
|
s.Add(&PeerConn{peer: peer{RemoteAddr: addrs[0], outgoing: true}})
|
2020-04-01 04:14:43 +08:00
|
|
|
targ := &pexState{
|
|
|
|
ev: []pexEvent{
|
|
|
|
pexEvent{pexAdd, addrs[0], pp.PexOutgoingConn},
|
|
|
|
},
|
|
|
|
nc: 1,
|
|
|
|
}
|
|
|
|
require.EqualValues(t, targ, s)
|
|
|
|
})
|
|
|
|
t.Run("belowTarg", func(t *testing.T) {
|
|
|
|
s := &pexState{
|
|
|
|
hold: []pexEvent{
|
|
|
|
pexEvent{pexDrop, addrs[1], 0},
|
|
|
|
},
|
|
|
|
nc: 0,
|
|
|
|
}
|
2020-07-15 14:15:38 +08:00
|
|
|
s.Add(&PeerConn{peer: peer{RemoteAddr: addrs[0]}})
|
2020-04-01 04:14:43 +08:00
|
|
|
targ := &pexState{
|
|
|
|
hold: []pexEvent{
|
|
|
|
pexEvent{pexDrop, addrs[1], 0},
|
|
|
|
},
|
|
|
|
ev: []pexEvent{
|
|
|
|
pexEvent{pexAdd, addrs[0], 0},
|
|
|
|
},
|
|
|
|
nc: 1,
|
|
|
|
}
|
|
|
|
require.EqualValues(t, targ, s)
|
|
|
|
})
|
|
|
|
t.Run("aboveTarg", func(t *testing.T) {
|
|
|
|
holdAddr := &net.TCPAddr{IP: net.IPv6loopback, Port: 4848}
|
|
|
|
s := &pexState{
|
|
|
|
hold: []pexEvent{
|
|
|
|
pexEvent{pexDrop, holdAddr, 0},
|
|
|
|
},
|
|
|
|
nc: pexTargAdded,
|
|
|
|
}
|
2020-07-15 14:15:38 +08:00
|
|
|
s.Add(&PeerConn{peer: peer{RemoteAddr: addrs[0]}})
|
2020-04-01 04:14:43 +08:00
|
|
|
targ := &pexState{
|
|
|
|
hold: []pexEvent{},
|
|
|
|
ev: []pexEvent{
|
|
|
|
pexEvent{pexDrop, holdAddr, 0},
|
|
|
|
pexEvent{pexAdd, addrs[0], 0},
|
|
|
|
},
|
|
|
|
nc: pexTargAdded + 1,
|
|
|
|
}
|
|
|
|
require.EqualValues(t, targ, s)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestPexDropped(t *testing.T) {
|
|
|
|
t.Run("belowTarg", func(t *testing.T) {
|
|
|
|
s := &pexState{nc: 1}
|
2020-10-13 10:33:22 +08:00
|
|
|
s.Drop(&PeerConn{peer: peer{RemoteAddr: addrs[0]}, pex: pexConnState{Listed: true}})
|
2020-04-01 04:14:43 +08:00
|
|
|
targ := &pexState{
|
|
|
|
hold: []pexEvent{pexEvent{pexDrop, addrs[0], 0}},
|
|
|
|
nc: 0,
|
|
|
|
}
|
|
|
|
require.EqualValues(t, targ, s)
|
|
|
|
})
|
|
|
|
t.Run("aboveTarg", func(t *testing.T) {
|
|
|
|
s := &pexState{nc: pexTargAdded + 1}
|
2020-10-13 10:33:22 +08:00
|
|
|
s.Drop(&PeerConn{peer: peer{RemoteAddr: addrs[0]}, pex: pexConnState{Listed: true}})
|
2020-04-01 04:14:43 +08:00
|
|
|
targ := &pexState{
|
|
|
|
ev: []pexEvent{pexEvent{pexDrop, addrs[0], 0}},
|
|
|
|
nc: pexTargAdded,
|
|
|
|
}
|
|
|
|
require.EqualValues(t, targ, s)
|
|
|
|
})
|
2020-04-09 00:03:29 +08:00
|
|
|
t.Run("aboveTargNotListed", func(t *testing.T) {
|
|
|
|
s := &pexState{nc: pexTargAdded + 1}
|
2020-10-13 10:33:22 +08:00
|
|
|
s.Drop(&PeerConn{peer: peer{RemoteAddr: addrs[0]}, pex: pexConnState{Listed: false}})
|
2020-04-09 00:03:29 +08:00
|
|
|
targ := &pexState{nc: pexTargAdded + 1}
|
|
|
|
require.EqualValues(t, targ, s)
|
|
|
|
})
|
2020-04-01 04:14:43 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestPexReset(t *testing.T) {
|
|
|
|
s := &pexState{
|
|
|
|
hold: []pexEvent{pexEvent{pexDrop, addrs[0], 0}},
|
|
|
|
ev: []pexEvent{pexEvent{pexAdd, addrs[1], 0}},
|
|
|
|
nc: 1,
|
|
|
|
}
|
|
|
|
s.Reset()
|
|
|
|
targ := new(pexState)
|
|
|
|
require.EqualValues(t, targ, s)
|
|
|
|
}
|
|
|
|
|
|
|
|
var testcases = []struct {
|
|
|
|
name string
|
|
|
|
in *pexState
|
|
|
|
arg int
|
2020-10-23 05:58:55 +08:00
|
|
|
targM pp.PexMsg
|
2020-04-01 04:14:43 +08:00
|
|
|
targS int
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
name: "empty",
|
|
|
|
in: &pexState{},
|
|
|
|
arg: 0,
|
2020-10-23 05:58:55 +08:00
|
|
|
targM: pp.PexMsg{},
|
2020-04-01 04:14:43 +08:00
|
|
|
targS: 0,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "add4",
|
|
|
|
in: &pexState{
|
|
|
|
ev: []pexEvent{
|
|
|
|
pexEvent{pexAdd, addrs[0], f},
|
|
|
|
pexEvent{pexAdd, addrs[1], f},
|
|
|
|
pexEvent{pexAdd, addrs[2], f},
|
|
|
|
pexEvent{pexAdd, addrs[3], f},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
arg: 0,
|
2020-10-23 05:58:55 +08:00
|
|
|
targM: pp.PexMsg{
|
2020-04-01 04:14:43 +08:00
|
|
|
Added: krpc.CompactIPv4NodeAddrs{
|
2020-10-23 05:58:55 +08:00
|
|
|
mustNodeAddr(addrs[2]),
|
|
|
|
mustNodeAddr(addrs[3]),
|
2020-04-01 04:14:43 +08:00
|
|
|
},
|
|
|
|
AddedFlags: []pp.PexPeerFlags{f, f},
|
|
|
|
Added6: krpc.CompactIPv6NodeAddrs{
|
2020-10-23 05:58:55 +08:00
|
|
|
mustNodeAddr(addrs[0]),
|
|
|
|
mustNodeAddr(addrs[1]),
|
2020-04-01 04:14:43 +08:00
|
|
|
},
|
|
|
|
Added6Flags: []pp.PexPeerFlags{f, f},
|
|
|
|
},
|
|
|
|
targS: 4,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "drop2",
|
|
|
|
arg: 0,
|
|
|
|
in: &pexState{
|
|
|
|
ev: []pexEvent{
|
|
|
|
pexEvent{pexDrop, addrs[0], f},
|
|
|
|
pexEvent{pexDrop, addrs[2], f},
|
|
|
|
},
|
|
|
|
},
|
2020-10-23 05:58:55 +08:00
|
|
|
targM: pp.PexMsg{
|
2020-04-01 04:14:43 +08:00
|
|
|
Dropped: krpc.CompactIPv4NodeAddrs{
|
2020-10-23 05:58:55 +08:00
|
|
|
mustNodeAddr(addrs[2]),
|
2020-04-01 04:14:43 +08:00
|
|
|
},
|
|
|
|
Dropped6: krpc.CompactIPv6NodeAddrs{
|
2020-10-23 05:58:55 +08:00
|
|
|
mustNodeAddr(addrs[0]),
|
2020-04-01 04:14:43 +08:00
|
|
|
},
|
|
|
|
},
|
|
|
|
targS: 2,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "add2drop1",
|
|
|
|
arg: 0,
|
|
|
|
in: &pexState{
|
|
|
|
ev: []pexEvent{
|
|
|
|
pexEvent{pexAdd, addrs[0], f},
|
|
|
|
pexEvent{pexAdd, addrs[1], f},
|
|
|
|
pexEvent{pexDrop, addrs[0], f},
|
|
|
|
},
|
|
|
|
},
|
2020-10-23 05:58:55 +08:00
|
|
|
targM: pp.PexMsg{
|
2020-04-01 04:14:43 +08:00
|
|
|
Added6: krpc.CompactIPv6NodeAddrs{
|
2020-10-23 05:58:55 +08:00
|
|
|
mustNodeAddr(addrs[1]),
|
2020-04-01 04:14:43 +08:00
|
|
|
},
|
|
|
|
Added6Flags: []pp.PexPeerFlags{f},
|
|
|
|
},
|
|
|
|
targS: 3,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "delayed",
|
|
|
|
arg: 0,
|
|
|
|
in: &pexState{
|
|
|
|
ev: []pexEvent{
|
|
|
|
pexEvent{pexAdd, addrs[0], f},
|
|
|
|
pexEvent{pexAdd, addrs[1], f},
|
|
|
|
pexEvent{pexAdd, addrs[2], f},
|
|
|
|
},
|
|
|
|
hold: []pexEvent{
|
|
|
|
pexEvent{pexDrop, addrs[0], f},
|
|
|
|
pexEvent{pexDrop, addrs[2], f},
|
|
|
|
pexEvent{pexDrop, addrs[1], f},
|
|
|
|
},
|
|
|
|
},
|
2020-10-23 05:58:55 +08:00
|
|
|
targM: pp.PexMsg{
|
2020-04-01 04:14:43 +08:00
|
|
|
Added: krpc.CompactIPv4NodeAddrs{
|
2020-10-23 05:58:55 +08:00
|
|
|
mustNodeAddr(addrs[2]),
|
2020-04-01 04:14:43 +08:00
|
|
|
},
|
|
|
|
AddedFlags: []pp.PexPeerFlags{f},
|
|
|
|
Added6: krpc.CompactIPv6NodeAddrs{
|
2020-10-23 05:58:55 +08:00
|
|
|
mustNodeAddr(addrs[0]),
|
|
|
|
mustNodeAddr(addrs[1]),
|
2020-04-01 04:14:43 +08:00
|
|
|
},
|
|
|
|
Added6Flags: []pp.PexPeerFlags{f, f},
|
|
|
|
},
|
|
|
|
targS: 3,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "followup",
|
|
|
|
arg: 1,
|
|
|
|
in: &pexState{
|
|
|
|
ev: []pexEvent{
|
|
|
|
pexEvent{pexAdd, addrs[0], f},
|
|
|
|
pexEvent{pexAdd, addrs[1], f},
|
|
|
|
},
|
|
|
|
},
|
2020-10-23 05:58:55 +08:00
|
|
|
targM: pp.PexMsg{
|
2020-04-01 04:14:43 +08:00
|
|
|
Added6: krpc.CompactIPv6NodeAddrs{
|
2020-10-23 05:58:55 +08:00
|
|
|
mustNodeAddr(addrs[1]),
|
2020-04-01 04:14:43 +08:00
|
|
|
},
|
|
|
|
Added6Flags: []pp.PexPeerFlags{f},
|
|
|
|
},
|
|
|
|
targS: 2,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2020-10-23 05:58:55 +08:00
|
|
|
// Represents the contents of a PexMsg in a way that supports equivalence checking in tests. This is
|
|
|
|
// necessary because pexMsgFactory uses maps and so ordering of the resultant PexMsg isn't
|
|
|
|
// deterministic. Because the flags are in a different array, we can't just use testify's
|
|
|
|
// ElementsMatch because the ordering *does* still matter between an added addr and its flags.
|
|
|
|
type comparablePexMsg struct {
|
|
|
|
added, added6 []pexMsgAdded
|
|
|
|
dropped, dropped6 []krpc.NodeAddr
|
|
|
|
}
|
|
|
|
|
|
|
|
func (me *comparablePexMsg) makeAdded(addrs []krpc.NodeAddr, flags []pp.PexPeerFlags) (ret []pexMsgAdded) {
|
|
|
|
for i, addr := range addrs {
|
|
|
|
ret = append(ret, pexMsgAdded{
|
|
|
|
NodeAddr: addr,
|
|
|
|
PexPeerFlags: flags[i],
|
|
|
|
})
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Such Rust-inspired.
|
|
|
|
func (me *comparablePexMsg) From(f pp.PexMsg) {
|
|
|
|
me.added = me.makeAdded(f.Added, f.AddedFlags)
|
|
|
|
me.added6 = me.makeAdded(f.Added6, f.Added6Flags)
|
|
|
|
me.dropped = f.Dropped
|
|
|
|
me.dropped6 = f.Dropped6
|
|
|
|
}
|
|
|
|
|
|
|
|
// For PexMsg created by pexMsgFactory, this is as good as it can get without using data structures
|
|
|
|
// in pexMsgFactory that preserve insert ordering.
|
|
|
|
func (actual comparablePexMsg) AssertEqual(t *testing.T, expected comparablePexMsg) {
|
|
|
|
assert.ElementsMatch(t, expected.added, actual.added)
|
|
|
|
assert.ElementsMatch(t, expected.added6, actual.added6)
|
|
|
|
assert.ElementsMatch(t, expected.dropped, actual.dropped)
|
|
|
|
assert.ElementsMatch(t, expected.dropped6, actual.dropped6)
|
|
|
|
}
|
|
|
|
|
|
|
|
func assertPexMsgsEqual(t *testing.T, expected, actual pp.PexMsg) {
|
|
|
|
var ec, ac comparablePexMsg
|
|
|
|
ec.From(expected)
|
|
|
|
ac.From(actual)
|
|
|
|
ac.AssertEqual(t, ec)
|
|
|
|
}
|
|
|
|
|
2020-04-01 04:14:43 +08:00
|
|
|
func TestPexGenmsg(t *testing.T) {
|
|
|
|
for _, tc := range testcases {
|
|
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
|
|
s := tc.in
|
|
|
|
m, seen := s.Genmsg(tc.arg)
|
2020-10-23 05:58:55 +08:00
|
|
|
assertPexMsgsEqual(t, tc.targM, m)
|
2020-04-01 04:14:43 +08:00
|
|
|
require.EqualValues(t, tc.targS, seen)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
2020-10-23 05:58:55 +08:00
|
|
|
|
|
|
|
func TestPexAdd(t *testing.T) {
|
|
|
|
addrs4 := []krpc.NodeAddr{
|
|
|
|
krpc.NodeAddr{IP: net.IPv4(127, 0, 0, 1), Port: 4747}, // 0
|
|
|
|
krpc.NodeAddr{IP: net.IPv4(127, 0, 0, 1), Port: 4748}, // 1
|
|
|
|
krpc.NodeAddr{IP: net.IPv4(127, 0, 0, 2), Port: 4747}, // 2
|
|
|
|
krpc.NodeAddr{IP: net.IPv4(127, 0, 0, 2), Port: 4748}, // 3
|
|
|
|
}
|
|
|
|
addrs6 := []krpc.NodeAddr{
|
|
|
|
krpc.NodeAddr{IP: net.IPv6loopback, Port: 4747}, // 0
|
|
|
|
krpc.NodeAddr{IP: net.IPv6loopback, Port: 4748}, // 1
|
|
|
|
krpc.NodeAddr{IP: net.IPv6loopback, Port: 4749}, // 2
|
|
|
|
krpc.NodeAddr{IP: net.IPv6loopback, Port: 4750}, // 3
|
|
|
|
}
|
|
|
|
f := pp.PexPrefersEncryption | pp.PexOutgoingConn
|
|
|
|
|
|
|
|
t.Run("ipv4", func(t *testing.T) {
|
|
|
|
addrs := addrs4
|
|
|
|
var m pexMsgFactory
|
|
|
|
m.Drop(addrs[0])
|
|
|
|
m.Add(addrs[1], f)
|
|
|
|
for _, addr := range addrs {
|
|
|
|
m.Add(addr, f)
|
|
|
|
}
|
|
|
|
targ := pp.PexMsg{
|
|
|
|
Added: krpc.CompactIPv4NodeAddrs{
|
|
|
|
addrs[1],
|
|
|
|
addrs[2],
|
|
|
|
addrs[3],
|
|
|
|
},
|
|
|
|
AddedFlags: []pp.PexPeerFlags{f, f, f},
|
|
|
|
}
|
|
|
|
assertPexMsgsEqual(t, targ, m.PexMsg())
|
|
|
|
})
|
|
|
|
t.Run("ipv6", func(t *testing.T) {
|
|
|
|
addrs := addrs6
|
|
|
|
var m pexMsgFactory
|
|
|
|
m.Drop(addrs[0])
|
|
|
|
m.Add(addrs[1], f)
|
|
|
|
for _, addr := range addrs {
|
|
|
|
m.Add(addr, f)
|
|
|
|
}
|
|
|
|
targ := pp.PexMsg{
|
|
|
|
Added6: krpc.CompactIPv6NodeAddrs{
|
|
|
|
addrs[1],
|
|
|
|
addrs[2],
|
|
|
|
addrs[3],
|
|
|
|
},
|
|
|
|
Added6Flags: []pp.PexPeerFlags{f, f, f},
|
|
|
|
}
|
|
|
|
assertPexMsgsEqual(t, targ, m.PexMsg())
|
|
|
|
})
|
|
|
|
t.Run("empty", func(t *testing.T) {
|
|
|
|
addr := krpc.NodeAddr{}
|
|
|
|
var xm pexMsgFactory
|
|
|
|
assert.Panics(t, func() { xm.Add(addr, f) })
|
|
|
|
m := xm.PexMsg()
|
|
|
|
require.EqualValues(t, 0, len(m.Added))
|
|
|
|
require.EqualValues(t, 0, len(m.AddedFlags))
|
|
|
|
require.EqualValues(t, 0, len(m.Added6))
|
|
|
|
require.EqualValues(t, 0, len(m.Added6Flags))
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestPexDrop(t *testing.T) {
|
|
|
|
addrs4 := []krpc.NodeAddr{
|
|
|
|
krpc.NodeAddr{IP: net.IPv4(127, 0, 0, 1), Port: 4747}, // 0
|
|
|
|
krpc.NodeAddr{IP: net.IPv4(127, 0, 0, 1), Port: 4748}, // 1
|
|
|
|
krpc.NodeAddr{IP: net.IPv4(127, 0, 0, 2), Port: 4747}, // 2
|
|
|
|
krpc.NodeAddr{IP: net.IPv4(127, 0, 0, 2), Port: 4748}, // 3
|
|
|
|
}
|
|
|
|
addrs6 := []krpc.NodeAddr{
|
|
|
|
krpc.NodeAddr{IP: net.IPv6loopback, Port: 4747}, // 0
|
|
|
|
krpc.NodeAddr{IP: net.IPv6loopback, Port: 4748}, // 1
|
|
|
|
krpc.NodeAddr{IP: net.IPv6loopback, Port: 4749}, // 2
|
|
|
|
krpc.NodeAddr{IP: net.IPv6loopback, Port: 4750}, // 3
|
|
|
|
}
|
|
|
|
f := pp.PexPrefersEncryption | pp.PexOutgoingConn
|
|
|
|
|
|
|
|
t.Run("ipv4", func(t *testing.T) {
|
|
|
|
addrs := addrs4
|
|
|
|
var m pexMsgFactory
|
|
|
|
m.Add(addrs[0], f)
|
|
|
|
m.Drop(addrs[1])
|
|
|
|
for _, addr := range addrs {
|
|
|
|
m.Drop(addr)
|
|
|
|
}
|
|
|
|
targ := pp.PexMsg{
|
|
|
|
Dropped: krpc.CompactIPv4NodeAddrs{
|
|
|
|
addrs[1],
|
|
|
|
addrs[2],
|
|
|
|
addrs[3],
|
|
|
|
},
|
|
|
|
}
|
|
|
|
assertPexMsgsEqual(t, targ, m.PexMsg())
|
|
|
|
})
|
|
|
|
t.Run("ipv6", func(t *testing.T) {
|
|
|
|
addrs := addrs6
|
|
|
|
var m pexMsgFactory
|
|
|
|
m.Add(addrs[0], f)
|
|
|
|
m.Drop(addrs[1])
|
|
|
|
for _, addr := range addrs {
|
|
|
|
m.Drop(addr)
|
|
|
|
}
|
|
|
|
targ := pp.PexMsg{
|
|
|
|
Dropped6: krpc.CompactIPv6NodeAddrs{
|
|
|
|
addrs[1],
|
|
|
|
addrs[2],
|
|
|
|
addrs[3],
|
|
|
|
},
|
|
|
|
}
|
|
|
|
assertPexMsgsEqual(t, targ, m.PexMsg())
|
|
|
|
})
|
|
|
|
t.Run("empty", func(t *testing.T) {
|
|
|
|
addr := krpc.NodeAddr{}
|
|
|
|
var xm pexMsgFactory
|
|
|
|
require.Panics(t, func() { xm.Drop(addr) })
|
|
|
|
m := xm.PexMsg()
|
|
|
|
require.EqualValues(t, 0, len(m.Dropped))
|
|
|
|
require.EqualValues(t, 0, len(m.Dropped6))
|
|
|
|
})
|
|
|
|
}
|