// 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 }