You've already forked torrent-client
mirror of
https://github.com/veggiedefender/torrent-client.git
synced 2025-11-06 09:29:16 +02:00
80 lines
1.7 KiB
Go
80 lines
1.7 KiB
Go
|
|
package p2p
|
||
|
|
|
||
|
|
import (
|
||
|
|
"fmt"
|
||
|
|
"net"
|
||
|
|
"strconv"
|
||
|
|
"sync"
|
||
|
|
"time"
|
||
|
|
|
||
|
|
"github.com/veggiedefender/torrent-client/message"
|
||
|
|
|
||
|
|
"github.com/veggiedefender/torrent-client/handshake"
|
||
|
|
)
|
||
|
|
|
||
|
|
type client struct {
|
||
|
|
conn net.Conn
|
||
|
|
bitfield message.Bitfield
|
||
|
|
Choked bool
|
||
|
|
Mux sync.Mutex
|
||
|
|
}
|
||
|
|
|
||
|
|
func completeHandshake(conn net.Conn, infohash, peerID [20]byte) (*handshake.Handshake, error) {
|
||
|
|
conn.SetDeadline(time.Now().Local().Add(3 * time.Second))
|
||
|
|
defer conn.SetDeadline(time.Time{}) // Disable the deadline
|
||
|
|
|
||
|
|
req := handshake.New(infohash, peerID)
|
||
|
|
_, err := conn.Write(req.Serialize())
|
||
|
|
if err != nil {
|
||
|
|
return nil, err
|
||
|
|
}
|
||
|
|
|
||
|
|
res, err := handshake.Read(conn)
|
||
|
|
if err != nil {
|
||
|
|
return nil, err
|
||
|
|
}
|
||
|
|
return res, nil
|
||
|
|
}
|
||
|
|
|
||
|
|
func recvBitfield(conn net.Conn) (message.Bitfield, error) {
|
||
|
|
conn.SetDeadline(time.Now().Local().Add(5 * time.Second))
|
||
|
|
defer conn.SetDeadline(time.Time{}) // Disable the deadline
|
||
|
|
|
||
|
|
msg, err := message.Read(conn)
|
||
|
|
if err != nil {
|
||
|
|
return nil, err
|
||
|
|
}
|
||
|
|
if msg.ID != message.MsgBitfield {
|
||
|
|
err := fmt.Errorf("Expected bitfield but got ID %d", msg.ID)
|
||
|
|
return nil, err
|
||
|
|
}
|
||
|
|
|
||
|
|
return msg.Payload, nil
|
||
|
|
}
|
||
|
|
|
||
|
|
func newClient(peer Peer, peerID, infoHash [20]byte) (*client, error) {
|
||
|
|
hostPort := net.JoinHostPort(peer.IP.String(), strconv.Itoa(int(peer.Port)))
|
||
|
|
conn, err := net.DialTimeout("tcp", hostPort, 3*time.Second)
|
||
|
|
if err != nil {
|
||
|
|
return nil, err
|
||
|
|
}
|
||
|
|
_, err = completeHandshake(conn, infoHash, peerID)
|
||
|
|
if err != nil {
|
||
|
|
return nil, err
|
||
|
|
}
|
||
|
|
bf, err := recvBitfield(conn)
|
||
|
|
if err != nil {
|
||
|
|
return nil, err
|
||
|
|
}
|
||
|
|
return &client{
|
||
|
|
conn: conn,
|
||
|
|
bitfield: bf,
|
||
|
|
Mux: sync.Mutex{},
|
||
|
|
Choked: true,
|
||
|
|
}, nil
|
||
|
|
}
|
||
|
|
|
||
|
|
func (c *client) hasPiece(index int) bool {
|
||
|
|
return c.bitfield.HasPiece(index)
|
||
|
|
}
|