2019-12-29 14:02:50 -05:00
|
|
|
package torrentfile
|
2019-12-21 23:14:33 -05:00
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"encoding/binary"
|
2020-01-02 10:30:30 -05:00
|
|
|
"fmt"
|
2019-12-21 23:14:33 -05:00
|
|
|
"net"
|
|
|
|
|
"net/http"
|
|
|
|
|
"net/url"
|
|
|
|
|
"strconv"
|
2020-01-02 10:34:47 -05:00
|
|
|
"time"
|
2019-12-21 23:14:33 -05:00
|
|
|
|
|
|
|
|
"github.com/jackpal/bencode-go"
|
2019-12-22 17:43:39 -05:00
|
|
|
"github.com/veggiedefender/torrent-client/p2p"
|
2019-12-21 23:14:33 -05:00
|
|
|
)
|
|
|
|
|
|
2019-12-22 14:54:54 -05:00
|
|
|
type bencodeTrackerResp struct {
|
2019-12-21 23:14:33 -05:00
|
|
|
Interval int `bencode:"interval"`
|
|
|
|
|
Peers string `bencode:"port"`
|
|
|
|
|
}
|
|
|
|
|
|
2019-12-22 17:43:39 -05:00
|
|
|
func parsePeers(peersBin string) ([]p2p.Peer, error) {
|
2020-01-02 10:28:58 -05:00
|
|
|
const peerSize = 6 // 4 for IP, 2 for port
|
2019-12-21 23:14:33 -05:00
|
|
|
numPeers := len(peersBin) / peerSize
|
|
|
|
|
if len(peersBin)%peerSize != 0 {
|
2020-01-02 10:30:30 -05:00
|
|
|
err := fmt.Errorf("Received malformed peers")
|
2019-12-21 23:14:33 -05:00
|
|
|
return nil, err
|
|
|
|
|
}
|
2019-12-22 17:43:39 -05:00
|
|
|
peers := make([]p2p.Peer, numPeers)
|
2019-12-21 23:14:33 -05:00
|
|
|
for i := 0; i < numPeers; i++ {
|
|
|
|
|
offset := i * peerSize
|
|
|
|
|
peers[i].IP = net.IP(peersBin[offset : offset+4])
|
|
|
|
|
peers[i].Port = binary.BigEndian.Uint16([]byte(peersBin[offset+4 : offset+6]))
|
|
|
|
|
}
|
|
|
|
|
return peers, nil
|
|
|
|
|
}
|
|
|
|
|
|
2019-12-29 14:02:50 -05:00
|
|
|
func (t *TorrentFile) buildTrackerURL(peerID [20]byte, port uint16) (string, error) {
|
2019-12-22 14:54:54 -05:00
|
|
|
base, err := url.Parse(t.Announce)
|
2019-12-21 23:14:33 -05:00
|
|
|
if err != nil {
|
|
|
|
|
return "", err
|
|
|
|
|
}
|
|
|
|
|
params := url.Values{
|
2019-12-22 15:19:46 -05:00
|
|
|
"info_hash": []string{string(t.InfoHash[:])},
|
|
|
|
|
"peer_id": []string{string(peerID[:])},
|
2019-12-22 14:54:54 -05:00
|
|
|
"port": []string{strconv.Itoa(int(Port))},
|
2019-12-21 23:14:33 -05:00
|
|
|
"uploaded": []string{"0"},
|
|
|
|
|
"downloaded": []string{"0"},
|
|
|
|
|
"compact": []string{"1"},
|
2019-12-22 14:54:54 -05:00
|
|
|
"left": []string{strconv.Itoa(t.Length)},
|
2019-12-21 23:14:33 -05:00
|
|
|
}
|
|
|
|
|
base.RawQuery = params.Encode()
|
|
|
|
|
return base.String(), nil
|
|
|
|
|
}
|
|
|
|
|
|
2020-01-02 10:32:20 -05:00
|
|
|
func (t *TorrentFile) requestPeers(peerID [20]byte, port uint16) ([]p2p.Peer, error) {
|
2019-12-22 14:54:54 -05:00
|
|
|
url, err := t.buildTrackerURL(peerID, port)
|
2019-12-21 23:14:33 -05:00
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
|
2020-01-02 10:34:47 -05:00
|
|
|
c := &http.Client{Timeout: 15 * time.Second}
|
|
|
|
|
resp, err := c.Get(url)
|
2019-12-21 23:14:33 -05:00
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
2019-12-22 14:54:54 -05:00
|
|
|
trackerResp := bencodeTrackerResp{}
|
2019-12-21 23:14:33 -05:00
|
|
|
err = bencode.Unmarshal(resp.Body, &trackerResp)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
peers, err := parsePeers(trackerResp.Peers)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return peers, nil
|
|
|
|
|
}
|