1
0
mirror of https://github.com/veggiedefender/torrent-client.git synced 2025-11-06 09:29:16 +02:00
Files
torrent-client/p2p/p2p.go

117 lines
2.5 KiB
Go
Raw Normal View History

2019-12-22 17:43:39 -05:00
package p2p
import (
2019-12-22 23:51:31 -05:00
"crypto/sha1"
"encoding/binary"
"encoding/hex"
"fmt"
"math"
2019-12-22 17:43:39 -05:00
"net"
"strconv"
2019-12-22 23:51:31 -05:00
"github.com/veggiedefender/torrent-client/message"
2019-12-22 17:43:39 -05:00
)
2019-12-22 22:25:57 -05:00
// Peer encodes connection information for a peer
2019-12-22 17:43:39 -05:00
type Peer struct {
IP net.IP
Port uint16
}
2019-12-22 22:25:57 -05:00
// Downloader holds data required to download a torrent from a list of peers
type Downloader struct {
2019-12-22 23:51:31 -05:00
Peers []Peer
PeerID [20]byte
InfoHash [20]byte
PieceHashes [][20]byte
Length int
2019-12-22 22:25:57 -05:00
}
2019-12-22 17:43:39 -05:00
2019-12-22 22:25:57 -05:00
// Download downloads a torrent
func (d *Downloader) Download() error {
2019-12-22 23:51:31 -05:00
conn, err := d.Peers[0].connect(d.PeerID, d.InfoHash)
defer conn.Close()
if err != nil {
return err
}
_, err = d.handshake(conn)
if err != nil {
return err
}
choked := false
pieceSize := d.Length / len(d.PieceHashes)
buf := make([]byte, pieceSize)
i := 0
for i < pieceSize {
msg, err := message.ReadMessage(conn)
if err != nil {
return err
}
switch msg.ID {
case message.MsgChoke:
choked = true
case message.MsgUnchoke:
choked = false
case message.MsgPiece:
begin := binary.BigEndian.Uint32(msg.Payload[4:8])
copy(buf[begin:], msg.Payload[8:])
i += (len(msg.Payload) - 8)
}
if !choked {
index := 0 // Piece number
begin := i // Offset
remain := pieceSize - i
length := int(math.Min(float64(16384), float64(pieceSize)))
length = int(math.Min(float64(remain), float64(length)))
payload := make([]byte, 12)
binary.BigEndian.PutUint32(payload[0:4], uint32(index))
binary.BigEndian.PutUint32(payload[4:8], uint32(begin))
binary.BigEndian.PutUint32(payload[8:12], uint32(length))
request := message.Message{
ID: message.MsgRequest,
Payload: payload,
}
_, err := conn.Write(request.Serialize())
if err != nil {
return err
}
}
}
s := sha1.Sum(buf)
fmt.Println(hex.EncodeToString(s[:]))
fmt.Println(hex.EncodeToString(d.PieceHashes[0][:]))
2019-12-22 22:25:57 -05:00
return nil
}
2019-12-22 17:43:39 -05:00
2019-12-22 23:51:31 -05:00
func (p *Peer) connect(peerID [20]byte, infoHash [20]byte) (net.Conn, error) {
2019-12-22 22:25:57 -05:00
hostPort := net.JoinHostPort(p.IP.String(), strconv.Itoa(int(p.Port)))
conn, err := net.Dial("tcp", hostPort)
2019-12-22 17:43:39 -05:00
if err != nil {
2019-12-22 22:25:57 -05:00
return nil, err
2019-12-22 17:43:39 -05:00
}
2019-12-22 22:25:57 -05:00
return conn, nil
2019-12-22 17:43:39 -05:00
}
2019-12-22 23:51:31 -05:00
func (d *Downloader) handshake(conn net.Conn) (*message.Handshake, error) {
req := message.Handshake{
Pstr: "BitTorrent protocol",
InfoHash: d.InfoHash,
PeerID: d.PeerID,
}
_, err := conn.Write(req.Serialize())
if err != nil {
return nil, err
}
res, err := message.ReadHandshake(conn)
if err != nil {
return nil, err
}
return res, nil
}