package tracker import ( "bytes" "crypto/rand" "encoding/binary" "fmt" "io" "io/ioutil" "net" "net/url" "strings" "sync" "testing" "github.com/stretchr/testify/assert" "github.com/stretchr/testify/require" "github.com/anacrolix/torrent/util" ) // Ensure net.IPs are stored big-endian, to match the way they're read from // the wire. func TestNetIPv4Bytes(t *testing.T) { ip := net.IP([]byte{127, 0, 0, 1}) if ip.String() != "127.0.0.1" { t.FailNow() } if string(ip) != "\x7f\x00\x00\x01" { t.Fatal([]byte(ip)) } } func TestMarshalAnnounceResponse(t *testing.T) { peers := util.CompactIPv4Peers{ {[]byte{127, 0, 0, 1}, 2}, {[]byte{255, 0, 0, 3}, 4}, } b, err := peers.MarshalBinary() require.NoError(t, err) require.EqualValues(t, "\x7f\x00\x00\x01\x00\x02\xff\x00\x00\x03\x00\x04", b) require.EqualValues(t, 12, binary.Size(AnnounceResponseHeader{})) } // Failure to write an entire packet to UDP is expected to given an error. func TestLongWriteUDP(t *testing.T) { l, err := net.ListenUDP("udp", nil) defer l.Close() if err != nil { t.Fatal(err) } c, err := net.DialUDP("udp", nil, l.LocalAddr().(*net.UDPAddr)) if err != nil { t.Fatal(err) } defer c.Close() for msgLen := 1; ; msgLen *= 2 { n, err := c.Write(make([]byte, msgLen)) if err != nil { require.Contains(t, err.Error(), "message too long") return } if n < msgLen { t.FailNow() } } } func TestShortBinaryRead(t *testing.T) { var data ResponseHeader err := binary.Read(bytes.NewBufferString("\x00\x00\x00\x01"), binary.BigEndian, &data) if err != io.ErrUnexpectedEOF { t.FailNow() } } func TestConvertInt16ToInt(t *testing.T) { i := 50000 if int(uint16(int16(i))) != 50000 { t.FailNow() } } func TestAnnounceLocalhost(t *testing.T) { srv := server{ t: map[[20]byte]torrent{ [20]byte{0xa3, 0x56, 0x41, 0x43, 0x74, 0x23, 0xe6, 0x26, 0xd9, 0x38, 0x25, 0x4a, 0x6b, 0x80, 0x49, 0x10, 0xa6, 0x67, 0xa, 0xc1}: { Seeders: 1, Leechers: 2, Peers: []util.CompactPeer{ {[]byte{1, 2, 3, 4}, 5}, {[]byte{6, 7, 8, 9}, 10}, }, }, }, } var err error srv.pc, err = net.ListenPacket("udp", ":0") require.NoError(t, err) defer srv.pc.Close() tr, err := New(fmt.Sprintf("udp://%s/announce", srv.pc.LocalAddr().String())) require.NoError(t, err) go func() { require.NoError(t, srv.serveOne()) }() err = tr.Connect() require.NoError(t, err) req := AnnounceRequest{ NumWant: -1, Event: Started, } rand.Read(req.PeerId[:]) copy(req.InfoHash[:], []uint8{0xa3, 0x56, 0x41, 0x43, 0x74, 0x23, 0xe6, 0x26, 0xd9, 0x38, 0x25, 0x4a, 0x6b, 0x80, 0x49, 0x10, 0xa6, 0x67, 0xa, 0xc1}) go func() { require.NoError(t, srv.serveOne()) }() ar, err := tr.Announce(&req) require.NoError(t, err) assert.EqualValues(t, 1, ar.Seeders) assert.EqualValues(t, 2, len(ar.Peers)) } func TestUDPTracker(t *testing.T) { tr, err := New("udp://tracker.openbittorrent.com:80/announce") require.NoError(t, err) if testing.Short() { t.SkipNow() } if err := tr.Connect(); err != nil { if strings.Contains(err.Error(), "no such host") { t.Skip(err) } if strings.Contains(err.Error(), "i/o timeout") { t.Skip(err) } t.Fatal(err) } req := AnnounceRequest{ NumWant: -1, // Event: Started, } rand.Read(req.PeerId[:]) copy(req.InfoHash[:], []uint8{0xa3, 0x56, 0x41, 0x43, 0x74, 0x23, 0xe6, 0x26, 0xd9, 0x38, 0x25, 0x4a, 0x6b, 0x80, 0x49, 0x10, 0xa6, 0x67, 0xa, 0xc1}) ar, err := tr.Announce(&req) if ne, ok := err.(net.Error); ok { if ne.Timeout() { t.Skip(err) } } require.NoError(t, err) t.Log(ar) } func TestAnnounceRandomInfoHashThirdParty(t *testing.T) { if testing.Short() { // This test involves contacting third party servers that may have // unpreditable results. t.SkipNow() } req := AnnounceRequest{ Event: Stopped, } rand.Read(req.PeerId[:]) rand.Read(req.InfoHash[:]) wg := sync.WaitGroup{} success := make(chan bool) fail := make(chan struct{}) for _, url := range []string{ "udp://tracker.openbittorrent.com:80/announce", "udp://tracker.publicbt.com:80", "udp://tracker.istole.it:6969", "udp://tracker.ccc.de:80", "udp://tracker.open.demonii.com:1337", "udp://open.demonii.com:1337", "udp://exodus.desync.com:6969", } { wg.Add(1) go func(url string) { defer wg.Done() tr, err := New(url) if err != nil { t.Fatal(err) } if err := tr.Connect(); err != nil { t.Log(err) return } resp, err := tr.Announce(&req) if err != nil { t.Logf("error announcing to %s: %s", url, err) return } if resp.Leechers != 0 || resp.Seeders != 0 || len(resp.Peers) != 0 { // The info hash we generated was random in 2^160 space. If we // get a hit, something is weird. t.Fatal(resp) } t.Logf("announced to %s", url) // TODO: Can probably get stuck here, but it's just a throwaway // test. success <- true }(url) } go func() { wg.Wait() close(fail) }() // Bail as quickly as we can. select { case <-fail: t.FailNow() case <-success: } } // Check that URLPath option is done correctly. func TestURLPathOption(t *testing.T) { conn, err := net.ListenUDP("udp", nil) if err != nil { panic(err) } defer conn.Close() cl := newClient(&url.URL{ Host: conn.LocalAddr().String(), Path: "/announce", }) go func() { err = cl.Connect() if err != nil { t.Fatal(err) } _, err = cl.Announce(&AnnounceRequest{}) if err != nil { t.Fatal(err) } }() var b [512]byte _, addr, _ := conn.ReadFrom(b[:]) r := bytes.NewReader(b[:]) var h RequestHeader read(r, &h) w := &bytes.Buffer{} write(w, ResponseHeader{ TransactionId: h.TransactionId, }) write(w, ConnectionResponse{42}) conn.WriteTo(w.Bytes(), addr) n, _, _ := conn.ReadFrom(b[:]) r = bytes.NewReader(b[:n]) read(r, &h) read(r, &AnnounceRequest{}) all, _ := ioutil.ReadAll(r) if string(all) != "\x02\x09/announce" { t.FailNow() } w = &bytes.Buffer{} write(w, ResponseHeader{ TransactionId: h.TransactionId, }) write(w, AnnounceResponseHeader{}) conn.WriteTo(w.Bytes(), addr) }