195 lines
4.3 KiB
Go
195 lines
4.3 KiB
Go
// Copyright 2014 The Chihaya Authors. All rights reserved.
|
|
// Use of this source code is governed by the BSD 2-Clause license,
|
|
// which can be found in the LICENSE file.
|
|
|
|
package http
|
|
|
|
import (
|
|
"net/http"
|
|
"net/http/httptest"
|
|
"net/url"
|
|
"reflect"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/chihaya/bencode"
|
|
"github.com/chihaya/chihaya/config"
|
|
)
|
|
|
|
func TestPublicAnnounce(t *testing.T) {
|
|
srv, err := setupTracker(&config.DefaultConfig)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer srv.Close()
|
|
|
|
// Add one seeder.
|
|
peer := makePeerParams("peer1", true)
|
|
expected := makeResponse(1, 0, bencode.List{})
|
|
checkAnnounce(peer, expected, srv, t)
|
|
|
|
// Add another seeder.
|
|
peer = makePeerParams("peer2", true)
|
|
expected = makeResponse(2, 0, bencode.List{})
|
|
checkAnnounce(peer, expected, srv, t)
|
|
|
|
// Add a leecher.
|
|
peer = makePeerParams("peer3", false)
|
|
expected = makeResponse(2, 1, bencode.List{
|
|
makePeerResponse("peer1"),
|
|
makePeerResponse("peer2"),
|
|
})
|
|
checkAnnounce(peer, expected, srv, t)
|
|
|
|
// Remove seeder.
|
|
peer = makePeerParams("peer1", true)
|
|
peer["event"] = "stopped"
|
|
expected = makeResponse(1, 1, nil)
|
|
checkAnnounce(peer, expected, srv, t)
|
|
|
|
// Check seeders.
|
|
peer = makePeerParams("peer3", false)
|
|
expected = makeResponse(1, 1, bencode.List{
|
|
makePeerResponse("peer2"),
|
|
})
|
|
checkAnnounce(peer, expected, srv, t)
|
|
}
|
|
|
|
func TestTorrentPurging(t *testing.T) {
|
|
cfg := config.DefaultConfig
|
|
srv, err := setupTracker(&cfg)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer srv.Close()
|
|
|
|
torrentApiPath := srv.URL + "/torrents/" + url.QueryEscape(infoHash)
|
|
|
|
// Add one seeder.
|
|
peer := makePeerParams("peer1", true)
|
|
announce(peer, srv)
|
|
|
|
_, status, err := fetchPath(torrentApiPath)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
} else if status != http.StatusOK {
|
|
t.Fatalf("expected torrent to exist (got %s)", http.StatusText(status))
|
|
}
|
|
|
|
// Remove seeder.
|
|
peer = makePeerParams("peer1", true)
|
|
peer["event"] = "stopped"
|
|
announce(peer, srv)
|
|
|
|
_, status, err = fetchPath(torrentApiPath)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
} else if status != http.StatusNotFound {
|
|
t.Fatalf("expected torrent to have been purged (got %s)", http.StatusText(status))
|
|
}
|
|
}
|
|
|
|
func TestStalePeerPurging(t *testing.T) {
|
|
cfg := config.DefaultConfig
|
|
cfg.Announce = config.Duration{10 * time.Millisecond}
|
|
|
|
srv, err := setupTracker(&cfg)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer srv.Close()
|
|
|
|
torrentApiPath := srv.URL + "/torrents/" + url.QueryEscape(infoHash)
|
|
|
|
// Add one seeder.
|
|
peer := makePeerParams("peer1", true)
|
|
announce(peer, srv)
|
|
|
|
_, status, err := fetchPath(torrentApiPath)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
} else if status != http.StatusOK {
|
|
t.Fatalf("expected torrent to exist (got %s)", http.StatusText(status))
|
|
}
|
|
|
|
// Add a leecher.
|
|
peer = makePeerParams("peer2", false)
|
|
expected := makeResponse(1, 1, bencode.List{
|
|
makePeerResponse("peer1"),
|
|
})
|
|
expected["interval"] = int64(0)
|
|
checkAnnounce(peer, expected, srv, t)
|
|
|
|
// Let them both expire.
|
|
time.Sleep(30 * time.Millisecond)
|
|
|
|
_, status, err = fetchPath(torrentApiPath)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
} else if status != http.StatusNotFound {
|
|
t.Fatalf("expected torrent to have been purged (got %s)", http.StatusText(status))
|
|
}
|
|
}
|
|
|
|
func makePeerParams(id string, seed bool) params {
|
|
left := "1"
|
|
if seed {
|
|
left = "0"
|
|
}
|
|
|
|
return params{
|
|
"info_hash": infoHash,
|
|
"peer_id": id,
|
|
"port": "1234",
|
|
"uploaded": "0",
|
|
"downloaded": "0",
|
|
"left": left,
|
|
"compact": "0",
|
|
"numwant": "50",
|
|
}
|
|
}
|
|
|
|
func makePeerResponse(id string) bencode.Dict {
|
|
return bencode.Dict{
|
|
"ip": "127.0.0.1",
|
|
"peer id": id,
|
|
"port": int64(1234),
|
|
}
|
|
}
|
|
|
|
func makeResponse(seeders, leechers int64, peers bencode.List) bencode.Dict {
|
|
dict := bencode.Dict{
|
|
"complete": seeders,
|
|
"incomplete": leechers,
|
|
"interval": int64(1800),
|
|
"min interval": int64(900),
|
|
}
|
|
|
|
if peers != nil {
|
|
dict["peers"] = peers
|
|
}
|
|
return dict
|
|
}
|
|
|
|
func checkAnnounce(p params, expected interface{}, srv *httptest.Server, t *testing.T) bool {
|
|
body, err := announce(p, srv)
|
|
if err != nil {
|
|
t.Error(err)
|
|
return false
|
|
}
|
|
|
|
if e, ok := expected.(bencode.Dict); ok {
|
|
sortPeersInResponse(e)
|
|
}
|
|
|
|
got, err := bencode.Unmarshal(body)
|
|
if e, ok := got.(bencode.Dict); ok {
|
|
sortPeersInResponse(e)
|
|
}
|
|
|
|
if !reflect.DeepEqual(got, expected) {
|
|
t.Errorf("\ngot: %#v\nwanted: %#v", got, expected)
|
|
return false
|
|
}
|
|
return true
|
|
}
|