FedP2P/client_test.go

775 lines
21 KiB
Go
Raw Normal View History

package torrent
import (
"encoding/binary"
"fmt"
2015-06-02 22:16:38 +08:00
"io"
"os"
"path/filepath"
"reflect"
"testing"
2021-01-29 20:32:01 +08:00
"testing/iotest"
"time"
2014-08-21 16:24:19 +08:00
2021-01-29 20:32:01 +08:00
"github.com/frankban/quicktest"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
2019-08-10 16:46:07 +08:00
"github.com/anacrolix/dht/v2"
2021-06-23 15:24:50 +08:00
"github.com/anacrolix/missinggo/v2"
"github.com/anacrolix/missinggo/v2/filecache"
2019-08-21 18:58:40 +08:00
2015-04-29 22:31:34 +08:00
"github.com/anacrolix/torrent/bencode"
"github.com/anacrolix/torrent/internal/testutil"
"github.com/anacrolix/torrent/iplist"
2015-06-02 22:16:38 +08:00
"github.com/anacrolix/torrent/metainfo"
2016-03-28 17:38:30 +08:00
"github.com/anacrolix/torrent/storage"
)
2014-08-21 16:07:06 +08:00
func TestClientDefault(t *testing.T) {
cl, err := NewClient(TestingConfig(t))
require.NoError(t, err)
2021-10-07 10:31:08 +08:00
require.Empty(t, cl.Close())
}
2018-04-12 21:34:31 +08:00
func TestClientNilConfig(t *testing.T) {
// The default config will put crap in the working directory.
origDir, _ := os.Getwd()
defer os.Chdir(origDir)
os.Chdir(t.TempDir())
2018-04-12 21:34:31 +08:00
cl, err := NewClient(nil)
require.NoError(t, err)
2021-10-07 10:31:08 +08:00
require.Empty(t, cl.Close())
2018-04-12 21:34:31 +08:00
}
func TestAddDropTorrent(t *testing.T) {
cl, err := NewClient(TestingConfig(t))
2016-03-28 18:57:04 +08:00
require.NoError(t, err)
2015-03-08 14:28:14 +08:00
defer cl.Close()
dir, mi := testutil.GreetingTestTorrent()
defer os.RemoveAll(dir)
tt, new, err := cl.AddTorrentSpec(TorrentSpecFromMetaInfo(mi))
2016-03-28 18:57:04 +08:00
require.NoError(t, err)
assert.True(t, new)
2016-07-05 22:42:16 +08:00
tt.SetMaxEstablishedConns(0)
tt.SetMaxEstablishedConns(1)
tt.Drop()
}
func TestAddTorrentNoSupportedTrackerSchemes(t *testing.T) {
2018-01-09 14:26:46 +08:00
// TODO?
t.SkipNow()
}
func TestAddTorrentNoUsableURLs(t *testing.T) {
2018-01-09 14:26:46 +08:00
// TODO?
t.SkipNow()
}
func TestAddPeersToUnknownTorrent(t *testing.T) {
2018-01-09 14:26:46 +08:00
// TODO?
t.SkipNow()
}
func TestPieceHashSize(t *testing.T) {
2018-01-09 14:26:46 +08:00
assert.Equal(t, 20, pieceHash.Size())
}
func TestTorrentInitialState(t *testing.T) {
dir, mi := testutil.GreetingTestTorrent()
defer os.RemoveAll(dir)
var cl Client
cl.init(TestingConfig(t))
cl.initLogger()
2018-01-28 13:07:11 +08:00
tor := cl.newTorrent(
mi.HashInfoBytes(),
storage.NewFileWithCompletion(t.TempDir(), storage.NewMapPieceCompletion()),
2018-01-28 13:07:11 +08:00
)
tor.setChunkSize(2)
2018-07-25 11:41:50 +08:00
tor.cl.lock()
2021-08-30 09:48:34 +08:00
err := tor.setInfoBytesLocked(mi.InfoBytes)
2018-07-25 11:41:50 +08:00
tor.cl.unlock()
2016-03-28 17:38:30 +08:00
require.NoError(t, err)
require.Len(t, tor.pieces, 3)
2014-12-09 11:59:01 +08:00
tor.pendAllChunkSpecs(0)
2018-07-25 11:41:50 +08:00
tor.cl.lock()
assert.EqualValues(t, 3, tor.pieceNumPendingChunks(0))
2018-07-25 11:41:50 +08:00
tor.cl.unlock()
assert.EqualValues(t, ChunkSpec{4, 1}, chunkIndexSpec(2, tor.pieceLength(0), tor.chunkSize))
}
2014-06-29 17:07:43 +08:00
func TestReducedDialTimeout(t *testing.T) {
cfg := NewDefaultClientConfig()
for _, _case := range []struct {
Max time.Duration
HalfOpenLimit int
PendingPeers int
ExpectedReduced time.Duration
}{
{cfg.NominalDialTimeout, 40, 0, cfg.NominalDialTimeout},
{cfg.NominalDialTimeout, 40, 1, cfg.NominalDialTimeout},
{cfg.NominalDialTimeout, 40, 39, cfg.NominalDialTimeout},
{cfg.NominalDialTimeout, 40, 40, cfg.NominalDialTimeout / 2},
{cfg.NominalDialTimeout, 40, 80, cfg.NominalDialTimeout / 3},
{cfg.NominalDialTimeout, 40, 4000, cfg.NominalDialTimeout / 101},
} {
reduced := reducedDialTimeout(cfg.MinDialTimeout, _case.Max, _case.HalfOpenLimit, _case.PendingPeers)
2014-11-19 11:53:00 +08:00
expected := _case.ExpectedReduced
if expected < cfg.MinDialTimeout {
expected = cfg.MinDialTimeout
2014-11-19 11:53:00 +08:00
}
if reduced != expected {
t.Fatalf("expected %s, got %s", _case.ExpectedReduced, reduced)
}
}
}
func TestAddDropManyTorrents(t *testing.T) {
cl, err := NewClient(TestingConfig(t))
require.NoError(t, err)
2015-03-08 14:28:14 +08:00
defer cl.Close()
for i := 0; i < 1000; i += 1 {
var spec TorrentSpec
binary.PutVarint(spec.InfoHash[:], int64(i))
tt, new, err := cl.AddTorrentSpec(&spec)
assert.NoError(t, err)
assert.True(t, new)
defer tt.Drop()
}
}
2020-10-11 09:40:43 +08:00
func fileCachePieceResourceStorage(fc *filecache.Cache) storage.ClientImpl {
return storage.NewResourcePiecesOpts(
fc.AsResourceProvider(),
storage.ResourcePiecesOpts{
LeaveIncompleteChunks: true,
},
)
}
func TestMergingTrackersByAddingSpecs(t *testing.T) {
cl, err := NewClient(TestingConfig(t))
require.NoError(t, err)
defer cl.Close()
spec := TorrentSpec{}
T, new, _ := cl.AddTorrentSpec(&spec)
if !new {
t.FailNow()
}
spec.Trackers = [][]string{{"http://a"}, {"udp://b"}}
_, new, _ = cl.AddTorrentSpec(&spec)
assert.False(t, new)
assert.EqualValues(t, [][]string{{"http://a"}, {"udp://b"}}, T.metainfo.AnnounceList)
// Because trackers are disabled in TestingConfig.
assert.EqualValues(t, 0, len(T.trackerAnnouncers))
}
2015-06-02 22:16:38 +08:00
// We read from a piece which is marked completed, but is missing data.
func TestCompletedPieceWrongSize(t *testing.T) {
cfg := TestingConfig(t)
2016-03-28 17:38:30 +08:00
cfg.DefaultStorage = badStorage{}
cl, err := NewClient(cfg)
require.NoError(t, err)
2015-06-02 22:16:38 +08:00
defer cl.Close()
info := metainfo.Info{
PieceLength: 15,
Pieces: make([]byte, 20),
Files: []metainfo.FileInfo{
2016-11-22 11:01:09 +08:00
{Path: []string{"greeting"}, Length: 13},
2015-06-02 22:16:38 +08:00
},
2016-05-09 13:47:39 +08:00
}
b, err := bencode.Marshal(info)
2017-11-07 13:11:59 +08:00
require.NoError(t, err)
2016-05-09 13:47:39 +08:00
tt, new, err := cl.AddTorrentSpec(&TorrentSpec{
InfoBytes: b,
InfoHash: metainfo.HashBytes(b),
2015-06-02 22:16:38 +08:00
})
require.NoError(t, err)
defer tt.Drop()
assert.True(t, new)
2015-06-02 22:16:38 +08:00
r := tt.NewReader()
defer r.Close()
2021-01-29 20:32:01 +08:00
quicktest.Check(t, iotest.TestReader(r, []byte(testutil.GreetingFileContents)), quicktest.IsNil)
2015-06-02 22:16:38 +08:00
}
func BenchmarkAddLargeTorrent(b *testing.B) {
cfg := TestingConfig(b)
cfg.DisableTCP = true
cfg.DisableUTP = true
cl, err := NewClient(cfg)
require.NoError(b, err)
defer cl.Close()
b.ReportAllocs()
for i := 0; i < b.N; i += 1 {
t, err := cl.AddTorrentFromFile("testdata/bootstrap.dat.torrent")
if err != nil {
b.Fatal(err)
}
t.Drop()
}
}
2015-07-15 13:51:42 +08:00
func TestResponsive(t *testing.T) {
seederDataDir, mi := testutil.GreetingTestTorrent()
defer os.RemoveAll(seederDataDir)
cfg := TestingConfig(t)
2015-07-15 13:51:42 +08:00
cfg.Seed = true
cfg.DataDir = seederDataDir
seeder, err := NewClient(cfg)
2015-07-15 13:51:42 +08:00
require.Nil(t, err)
defer seeder.Close()
seederTorrent, _, _ := seeder.AddTorrentSpec(TorrentSpecFromMetaInfo(mi))
seederTorrent.VerifyData()
leecherDataDir := t.TempDir()
cfg = TestingConfig(t)
2015-07-15 13:51:42 +08:00
cfg.DataDir = leecherDataDir
leecher, err := NewClient(cfg)
2015-07-15 13:51:42 +08:00
require.Nil(t, err)
defer leecher.Close()
leecherTorrent, _, _ := leecher.AddTorrentSpec(func() (ret *TorrentSpec) {
ret = TorrentSpecFromMetaInfo(mi)
ret.ChunkSize = 2
return
}())
leecherTorrent.AddClientPeer(seeder)
2015-07-15 13:51:42 +08:00
reader := leecherTorrent.NewReader()
2016-02-04 22:17:26 +08:00
defer reader.Close()
2015-07-15 13:51:42 +08:00
reader.SetReadahead(0)
reader.SetResponsive()
b := make([]byte, 2)
2017-11-07 13:11:59 +08:00
_, err = reader.Seek(3, io.SeekStart)
require.NoError(t, err)
_, err = io.ReadFull(reader, b)
2015-07-15 13:51:42 +08:00
assert.Nil(t, err)
assert.EqualValues(t, "lo", string(b))
2017-11-07 13:11:59 +08:00
_, err = reader.Seek(11, io.SeekStart)
require.NoError(t, err)
n, err := io.ReadFull(reader, b)
2015-07-15 13:51:42 +08:00
assert.Nil(t, err)
assert.EqualValues(t, 2, n)
assert.EqualValues(t, "d\n", string(b))
}
func TestTorrentDroppedDuringResponsiveRead(t *testing.T) {
seederDataDir, mi := testutil.GreetingTestTorrent()
defer os.RemoveAll(seederDataDir)
cfg := TestingConfig(t)
cfg.Seed = true
cfg.DataDir = seederDataDir
seeder, err := NewClient(cfg)
require.Nil(t, err)
defer seeder.Close()
seederTorrent, _, _ := seeder.AddTorrentSpec(TorrentSpecFromMetaInfo(mi))
seederTorrent.VerifyData()
leecherDataDir := t.TempDir()
cfg = TestingConfig(t)
cfg.DataDir = leecherDataDir
leecher, err := NewClient(cfg)
require.Nil(t, err)
defer leecher.Close()
leecherTorrent, _, _ := leecher.AddTorrentSpec(func() (ret *TorrentSpec) {
ret = TorrentSpecFromMetaInfo(mi)
ret.ChunkSize = 2
return
}())
leecherTorrent.AddClientPeer(seeder)
reader := leecherTorrent.NewReader()
2016-02-04 22:17:26 +08:00
defer reader.Close()
reader.SetReadahead(0)
reader.SetResponsive()
b := make([]byte, 2)
2017-11-07 13:11:59 +08:00
_, err = reader.Seek(3, io.SeekStart)
require.NoError(t, err)
_, err = io.ReadFull(reader, b)
assert.Nil(t, err)
assert.EqualValues(t, "lo", string(b))
2017-11-07 13:11:59 +08:00
_, err = reader.Seek(11, io.SeekStart)
require.NoError(t, err)
leecherTorrent.Drop()
n, err := reader.Read(b)
assert.EqualError(t, err, "torrent closed")
assert.EqualValues(t, 0, n)
}
2020-04-16 12:52:52 +08:00
func TestDhtInheritBlocklist(t *testing.T) {
ipl := iplist.New(nil)
require.NotNil(t, ipl)
cfg := TestingConfig(t)
2016-01-16 21:12:53 +08:00
cfg.IPBlocklist = ipl
cfg.NoDHT = false
cl, err := NewClient(cfg)
require.NoError(t, err)
defer cl.Close()
numServers := 0
cl.eachDhtServer(func(s DhtServer) {
2020-04-16 12:52:52 +08:00
t.Log(s)
2021-05-24 15:37:04 +08:00
assert.Equal(t, ipl, s.(AnacrolixDhtServerWrapper).Server.IPBlocklist())
numServers++
})
assert.EqualValues(t, 2, numServers)
}
2015-08-23 10:50:32 +08:00
// Check that stuff is merged in subsequent AddTorrentSpec for the same
// infohash.
func TestAddTorrentSpecMerging(t *testing.T) {
cl, err := NewClient(TestingConfig(t))
2015-08-23 10:50:32 +08:00
require.NoError(t, err)
defer cl.Close()
dir, mi := testutil.GreetingTestTorrent()
defer os.RemoveAll(dir)
tt, new, err := cl.AddTorrentSpec(&TorrentSpec{
InfoHash: mi.HashInfoBytes(),
})
2015-08-23 10:50:32 +08:00
require.NoError(t, err)
require.True(t, new)
require.Nil(t, tt.Info())
_, new, err = cl.AddTorrentSpec(TorrentSpecFromMetaInfo(mi))
require.NoError(t, err)
require.False(t, new)
require.NotNil(t, tt.Info())
}
func TestTorrentDroppedBeforeGotInfo(t *testing.T) {
dir, mi := testutil.GreetingTestTorrent()
os.RemoveAll(dir)
cl, _ := NewClient(TestingConfig(t))
defer cl.Close()
tt, _, _ := cl.AddTorrentSpec(&TorrentSpec{
InfoHash: mi.HashInfoBytes(),
})
tt.Drop()
assert.EqualValues(t, 0, len(cl.Torrents()))
select {
case <-tt.GotInfo():
t.FailNow()
default:
}
}
func writeTorrentData(ts *storage.Torrent, info metainfo.Info, b []byte) {
for i := 0; i < info.NumPieces(); i += 1 {
p := info.Piece(i)
ts.Piece(p).WriteAt(b[p.Offset():p.Offset()+p.Length()], 0)
2016-03-29 08:14:34 +08:00
}
}
2020-10-11 09:40:43 +08:00
func testAddTorrentPriorPieceCompletion(t *testing.T, alreadyCompleted bool, csf func(*filecache.Cache) storage.ClientImpl) {
fileCacheDir := t.TempDir()
2016-03-29 08:14:34 +08:00
fileCache, err := filecache.NewCache(fileCacheDir)
require.NoError(t, err)
greetingDataTempDir, greetingMetainfo := testutil.GreetingTestTorrent()
defer os.RemoveAll(greetingDataTempDir)
filePieceStore := csf(fileCache)
info, err := greetingMetainfo.UnmarshalInfo()
require.NoError(t, err)
ih := greetingMetainfo.HashInfoBytes()
greetingData, err := storage.NewClient(filePieceStore).OpenTorrent(&info, ih)
2016-03-29 08:14:34 +08:00
require.NoError(t, err)
writeTorrentData(greetingData, info, []byte(testutil.GreetingFileContents))
2016-03-29 08:14:34 +08:00
// require.Equal(t, len(testutil.GreetingFileContents), written)
// require.NoError(t, err)
for i := 0; i < info.NumPieces(); i++ {
p := info.Piece(i)
2016-03-29 08:14:34 +08:00
if alreadyCompleted {
require.NoError(t, greetingData.Piece(p).MarkComplete())
2016-03-29 08:14:34 +08:00
}
}
cfg := TestingConfig(t)
2016-03-29 08:14:34 +08:00
// TODO: Disable network option?
cfg.DisableTCP = true
cfg.DisableUTP = true
cfg.DefaultStorage = filePieceStore
cl, err := NewClient(cfg)
2016-03-29 08:14:34 +08:00
require.NoError(t, err)
defer cl.Close()
tt, err := cl.AddTorrent(greetingMetainfo)
require.NoError(t, err)
psrs := tt.PieceStateRuns()
assert.Len(t, psrs, 1)
assert.EqualValues(t, 3, psrs[0].Length)
assert.Equal(t, alreadyCompleted, psrs[0].Complete)
if alreadyCompleted {
r := tt.NewReader()
2021-01-29 20:32:01 +08:00
quicktest.Check(t, iotest.TestReader(r, []byte(testutil.GreetingFileContents)), quicktest.IsNil)
2016-03-29 08:14:34 +08:00
}
}
func TestAddTorrentPiecesAlreadyCompleted(t *testing.T) {
testAddTorrentPriorPieceCompletion(t, true, fileCachePieceResourceStorage)
2016-03-29 08:14:34 +08:00
}
func TestAddTorrentPiecesNotAlreadyCompleted(t *testing.T) {
testAddTorrentPriorPieceCompletion(t, false, fileCachePieceResourceStorage)
2016-03-29 08:14:34 +08:00
}
func TestAddMetainfoWithNodes(t *testing.T) {
cfg := TestingConfig(t)
cfg.ListenHost = func(string) string { return "" }
cfg.NoDHT = false
2020-03-17 10:47:53 +08:00
cfg.DhtStartingNodes = func(string) dht.StartingNodesGetter { return func() ([]dht.Addr, error) { return nil, nil } }
// For now, we want to just jam the nodes into the table, without verifying them first. Also the
// DHT code doesn't support mixing secure and insecure nodes if security is enabled (yet).
// cfg.DHTConfig.NoSecurity = true
cl, err := NewClient(cfg)
require.NoError(t, err)
defer cl.Close()
2018-04-12 14:11:22 +08:00
sum := func() (ret int64) {
cl.eachDhtServer(func(s DhtServer) {
ret += s.Stats().(dht.ServerStats).OutboundQueriesAttempted
})
return
}
assert.EqualValues(t, 0, sum())
tt, err := cl.AddTorrentFromFile("metainfo/testdata/issue_65a.torrent")
require.NoError(t, err)
2017-07-20 22:40:49 +08:00
// Nodes are not added or exposed in Torrent's metainfo. We just randomly
// check if the announce-list is here instead. TODO: Add nodes.
assert.Len(t, tt.metainfo.AnnounceList, 5)
2017-07-20 22:40:49 +08:00
// There are 6 nodes in the torrent file.
2019-01-22 05:36:40 +08:00
for sum() != int64(6*len(cl.dhtServers)) {
time.Sleep(time.Millisecond)
}
}
2016-02-26 19:10:29 +08:00
type testDownloadCancelParams struct {
SetLeecherStorageCapacity bool
LeecherStorageCapacity int64
Cancel bool
}
func testDownloadCancel(t *testing.T, ps testDownloadCancelParams) {
greetingTempDir, mi := testutil.GreetingTestTorrent()
defer os.RemoveAll(greetingTempDir)
cfg := TestingConfig(t)
2016-02-26 19:10:29 +08:00
cfg.Seed = true
cfg.DataDir = greetingTempDir
seeder, err := NewClient(cfg)
2016-02-26 19:10:29 +08:00
require.NoError(t, err)
defer seeder.Close()
defer testutil.ExportStatusWriter(seeder, "s", t)()
seederTorrent, _, _ := seeder.AddTorrentSpec(TorrentSpecFromMetaInfo(mi))
seederTorrent.VerifyData()
leecherDataDir := t.TempDir()
2016-03-29 08:14:34 +08:00
fc, err := filecache.NewCache(leecherDataDir)
require.NoError(t, err)
if ps.SetLeecherStorageCapacity {
fc.SetCapacity(ps.LeecherStorageCapacity)
}
cfg.DefaultStorage = storage.NewResourcePieces(fc.AsResourceProvider())
2016-03-28 17:38:30 +08:00
cfg.DataDir = leecherDataDir
leecher, err := NewClient(cfg)
require.NoError(t, err)
2016-02-26 19:10:29 +08:00
defer leecher.Close()
defer testutil.ExportStatusWriter(leecher, "l", t)()
2016-02-26 19:10:29 +08:00
leecherGreeting, new, err := leecher.AddTorrentSpec(func() (ret *TorrentSpec) {
ret = TorrentSpecFromMetaInfo(mi)
ret.ChunkSize = 2
return
}())
require.NoError(t, err)
assert.True(t, new)
psc := leecherGreeting.SubscribePieceStateChanges()
defer psc.Close()
2018-01-27 11:31:31 +08:00
2018-07-25 11:41:50 +08:00
leecherGreeting.cl.lock()
2018-01-27 11:31:31 +08:00
leecherGreeting.downloadPiecesLocked(0, leecherGreeting.numPieces())
2016-02-26 19:10:29 +08:00
if ps.Cancel {
leecherGreeting.cancelPiecesLocked(0, leecherGreeting.NumPieces(), "")
2016-02-26 19:10:29 +08:00
}
2018-07-25 11:41:50 +08:00
leecherGreeting.cl.unlock()
done := make(chan struct{})
defer close(done)
go leecherGreeting.AddClientPeer(seeder)
2016-02-26 19:10:29 +08:00
completes := make(map[int]bool, 3)
expected := func() map[int]bool {
if ps.Cancel {
return map[int]bool{0: false, 1: false, 2: false}
} else {
return map[int]bool{0: true, 1: true, 2: true}
}
}()
for !reflect.DeepEqual(completes, expected) {
2018-11-28 07:32:44 +08:00
_v := <-psc.Values
v := _v.(PieceStateChange)
completes[v.Index] = v.Complete
2016-02-26 19:10:29 +08:00
}
}
func TestTorrentDownloadAll(t *testing.T) {
testDownloadCancel(t, testDownloadCancelParams{})
}
func TestTorrentDownloadAllThenCancel(t *testing.T) {
testDownloadCancel(t, testDownloadCancelParams{
Cancel: true,
})
}
// Ensure that it's an error for a peer to send an invalid have message.
func TestPeerInvalidHave(t *testing.T) {
cfg := TestingConfig(t)
cfg.DropMutuallyCompletePeers = false
cl, err := NewClient(cfg)
require.NoError(t, err)
defer cl.Close()
info := metainfo.Info{
PieceLength: 1,
Pieces: make([]byte, 20),
Files: []metainfo.FileInfo{{Length: 1}},
2016-05-09 13:47:39 +08:00
}
infoBytes, err := bencode.Marshal(info)
require.NoError(t, err)
2016-05-09 13:47:39 +08:00
tt, _new, err := cl.AddTorrentSpec(&TorrentSpec{
InfoBytes: infoBytes,
InfoHash: metainfo.HashBytes(infoBytes),
2017-09-18 10:15:14 +08:00
Storage: badStorage{},
})
require.NoError(t, err)
assert.True(t, _new)
defer tt.Drop()
2021-01-20 10:10:32 +08:00
cn := &PeerConn{Peer: Peer{
t: tt,
callbacks: &cfg.Callbacks,
2020-05-30 08:14:20 +08:00
}}
tt.conns[cn] = struct{}{}
cn.peerImpl = cn
cl.lock()
defer cl.unlock()
assert.NoError(t, cn.peerSentHave(0))
assert.Error(t, cn.peerSentHave(1))
}
2016-03-28 17:38:30 +08:00
func TestPieceCompletedInStorageButNotClient(t *testing.T) {
greetingTempDir, greetingMetainfo := testutil.GreetingTestTorrent()
defer os.RemoveAll(greetingTempDir)
cfg := TestingConfig(t)
2016-03-28 17:38:30 +08:00
cfg.DataDir = greetingTempDir
seeder, err := NewClient(TestingConfig(t))
2016-03-28 17:38:30 +08:00
require.NoError(t, err)
seeder.AddTorrentSpec(&TorrentSpec{
InfoBytes: greetingMetainfo.InfoBytes,
2016-03-28 17:38:30 +08:00
})
}
2016-05-24 17:35:23 +08:00
// Check that when the listen port is 0, all the protocols listened on have
// the same port, and it isn't zero.
func TestClientDynamicListenPortAllProtocols(t *testing.T) {
cl, err := NewClient(TestingConfig(t))
2016-05-24 17:35:23 +08:00
require.NoError(t, err)
defer cl.Close()
port := cl.LocalPort()
assert.NotEqual(t, 0, port)
cl.eachListener(func(s Listener) bool {
assert.Equal(t, port, missinggo.AddrPort(s.Addr()))
return true
})
2016-05-24 17:35:23 +08:00
}
func TestClientDynamicListenTCPOnly(t *testing.T) {
cfg := TestingConfig(t)
2016-05-24 17:35:23 +08:00
cfg.DisableUTP = true
cfg.DisableTCP = false
cl, err := NewClient(cfg)
2016-05-24 17:35:23 +08:00
require.NoError(t, err)
defer cl.Close()
assert.NotEqual(t, 0, cl.LocalPort())
2016-05-24 17:35:23 +08:00
}
func TestClientDynamicListenUTPOnly(t *testing.T) {
cfg := TestingConfig(t)
2016-05-24 17:35:23 +08:00
cfg.DisableTCP = true
cfg.DisableUTP = false
cl, err := NewClient(cfg)
2016-05-24 17:35:23 +08:00
require.NoError(t, err)
defer cl.Close()
assert.NotEqual(t, 0, cl.LocalPort())
}
func totalConns(tts []*Torrent) (ret int) {
for _, tt := range tts {
2018-07-25 11:41:50 +08:00
tt.cl.lock()
ret += len(tt.conns)
2018-07-25 11:41:50 +08:00
tt.cl.unlock()
}
return
}
func TestSetMaxEstablishedConn(t *testing.T) {
var tts []*Torrent
ih := testutil.GreetingMetaInfo().HashInfoBytes()
cfg := TestingConfig(t)
cfg.DisableAcceptRateLimiting = true
cfg.DropDuplicatePeerIds = true
for i := 0; i < 3; i += 1 {
cl, err := NewClient(cfg)
require.NoError(t, err)
defer cl.Close()
tt, _ := cl.AddTorrentInfoHash(ih)
tt.SetMaxEstablishedConns(2)
defer testutil.ExportStatusWriter(cl, fmt.Sprintf("%d", i), t)()
tts = append(tts, tt)
}
addPeers := func() {
for _, tt := range tts {
for _, _tt := range tts {
// if tt != _tt {
tt.AddClientPeer(_tt.cl)
// }
}
}
}
waitTotalConns := func(num int) {
for totalConns(tts) != num {
addPeers()
time.Sleep(time.Millisecond)
}
}
addPeers()
waitTotalConns(6)
tts[0].SetMaxEstablishedConns(1)
waitTotalConns(4)
tts[0].SetMaxEstablishedConns(0)
waitTotalConns(2)
tts[0].SetMaxEstablishedConns(1)
addPeers()
waitTotalConns(4)
tts[0].SetMaxEstablishedConns(2)
addPeers()
waitTotalConns(6)
}
2016-09-16 22:01:15 +08:00
2019-07-19 11:04:10 +08:00
// Creates a file containing its own name as data. Make a metainfo from that, adds it to the given
// client, and returns a magnet link.
2021-12-18 05:58:56 +08:00
func makeMagnet(t *testing.T, cl *Client, dir, name string) string {
2021-11-08 11:47:01 +08:00
os.MkdirAll(dir, 0o770)
file, err := os.Create(filepath.Join(dir, name))
2016-09-16 22:01:15 +08:00
require.NoError(t, err)
file.Write([]byte(name))
2016-09-20 16:39:36 +08:00
file.Close()
2016-09-16 22:01:15 +08:00
mi := metainfo.MetaInfo{}
mi.SetDefaults()
2016-09-20 16:39:36 +08:00
info := metainfo.Info{PieceLength: 256 * 1024}
err = info.BuildFromFilePath(filepath.Join(dir, name))
2016-09-16 22:01:15 +08:00
require.NoError(t, err)
mi.InfoBytes, err = bencode.Marshal(info)
require.NoError(t, err)
magnet := mi.Magnet(nil, &info).String()
2016-09-16 22:01:15 +08:00
tr, err := cl.AddTorrent(&mi)
require.NoError(t, err)
require.True(t, tr.Seeding())
tr.VerifyData()
2016-09-16 22:01:15 +08:00
return magnet
}
// https://github.com/anacrolix/torrent/issues/114
func TestMultipleTorrentsWithEncryption(t *testing.T) {
testSeederLeecherPair(
t,
func(cfg *ClientConfig) {
cfg.HeaderObfuscationPolicy.Preferred = true
cfg.HeaderObfuscationPolicy.RequirePreferred = true
},
func(cfg *ClientConfig) {
cfg.HeaderObfuscationPolicy.RequirePreferred = false
},
)
}
// Test that the leecher can download a torrent in its entirety from the seeder. Note that the
// seeder config is done first.
2021-12-18 05:58:56 +08:00
func testSeederLeecherPair(t *testing.T, seeder, leecher func(*ClientConfig)) {
cfg := TestingConfig(t)
2016-09-16 22:01:15 +08:00
cfg.Seed = true
cfg.DataDir = filepath.Join(cfg.DataDir, "server")
2021-11-08 11:47:01 +08:00
os.Mkdir(cfg.DataDir, 0o755)
seeder(cfg)
server, err := NewClient(cfg)
2016-09-16 22:01:15 +08:00
require.NoError(t, err)
defer server.Close()
defer testutil.ExportStatusWriter(server, "s", t)()
2016-09-16 22:01:15 +08:00
magnet1 := makeMagnet(t, server, cfg.DataDir, "test1")
// Extra torrents are added to test the seeder having to match incoming obfuscated headers
// against more than one torrent. See issue #114
2016-09-16 22:01:15 +08:00
makeMagnet(t, server, cfg.DataDir, "test2")
for i := 0; i < 100; i++ {
makeMagnet(t, server, cfg.DataDir, fmt.Sprintf("test%d", i+2))
}
cfg = TestingConfig(t)
cfg.DataDir = filepath.Join(cfg.DataDir, "client")
leecher(cfg)
client, err := NewClient(cfg)
2016-09-16 22:01:15 +08:00
require.NoError(t, err)
defer client.Close()
defer testutil.ExportStatusWriter(client, "c", t)()
2016-09-16 22:01:15 +08:00
tr, err := client.AddMagnet(magnet1)
require.NoError(t, err)
tr.AddClientPeer(server)
2016-09-20 16:39:36 +08:00
<-tr.GotInfo()
2016-09-16 22:01:15 +08:00
tr.DownloadAll()
client.WaitAll()
}
// This appears to be the situation with the S3 BitTorrent client.
func TestObfuscatedHeaderFallbackSeederDisallowsLeecherPrefers(t *testing.T) {
// Leecher prefers obfuscation, but the seeder does not allow it.
testSeederLeecherPair(
t,
func(cfg *ClientConfig) {
cfg.HeaderObfuscationPolicy.Preferred = false
cfg.HeaderObfuscationPolicy.RequirePreferred = true
},
func(cfg *ClientConfig) {
cfg.HeaderObfuscationPolicy.Preferred = true
cfg.HeaderObfuscationPolicy.RequirePreferred = false
},
)
}
func TestObfuscatedHeaderFallbackSeederRequiresLeecherPrefersNot(t *testing.T) {
// Leecher prefers no obfuscation, but the seeder enforces it.
testSeederLeecherPair(
t,
func(cfg *ClientConfig) {
cfg.HeaderObfuscationPolicy.Preferred = true
cfg.HeaderObfuscationPolicy.RequirePreferred = true
},
func(cfg *ClientConfig) {
cfg.HeaderObfuscationPolicy.Preferred = false
cfg.HeaderObfuscationPolicy.RequirePreferred = false
},
)
}
func TestClientAddressInUse(t *testing.T) {
s, _ := NewUtpSocket("udp", ":50007", nil)
if s != nil {
defer s.Close()
}
cfg := TestingConfig(t).SetListenAddr(":50007")
cl, err := NewClient(cfg)
require.Error(t, err)
require.Nil(t, cl)
}
func TestClientHasDhtServersWhenUtpDisabled(t *testing.T) {
cc := TestingConfig(t)
cc.DisableUTP = true
cc.NoDHT = false
cl, err := NewClient(cc)
require.NoError(t, err)
defer cl.Close()
assert.NotEmpty(t, cl.DhtServers())
}
2019-10-11 14:34:07 +08:00
func TestClientDisabledImplicitNetworksButDhtEnabled(t *testing.T) {
cfg := TestingConfig(t)
cfg.DisableTCP = true
cfg.DisableUTP = true
cfg.NoDHT = false
cl, err := NewClient(cfg)
require.NoError(t, err)
defer cl.Close()
assert.Empty(t, cl.listeners)
assert.NotEmpty(t, cl.DhtServers())
}