You've already forked torrent-client
mirror of
https://github.com/veggiedefender/torrent-client.git
synced 2025-11-06 09:29:16 +02:00
Add p2p client
This commit is contained in:
79
p2p/client.go
Normal file
79
p2p/client.go
Normal file
@@ -0,0 +1,79 @@
|
|||||||
|
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)
|
||||||
|
}
|
||||||
121
p2p/p2p.go
121
p2p/p2p.go
@@ -1,11 +1,9 @@
|
|||||||
package p2p
|
package p2p
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"fmt"
|
||||||
"net"
|
"net"
|
||||||
"strconv"
|
"sync"
|
||||||
"time"
|
|
||||||
|
|
||||||
"github.com/veggiedefender/torrent-client/handshake"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// Peer encodes connection information for a peer
|
// Peer encodes connection information for a peer
|
||||||
@@ -23,73 +21,66 @@ type Download struct {
|
|||||||
Length int
|
Length int
|
||||||
}
|
}
|
||||||
|
|
||||||
type peerState struct {
|
type pieceWork struct {
|
||||||
peer *Peer
|
index int
|
||||||
conn net.Conn
|
hash [20]byte
|
||||||
}
|
}
|
||||||
|
|
||||||
type swarm struct {
|
type swarm struct {
|
||||||
peerStates []*peerState
|
clients []*client
|
||||||
}
|
queue chan *pieceWork
|
||||||
|
mux sync.Mutex
|
||||||
func (p *Peer) connect(peerID [20]byte, infoHash [20]byte) (net.Conn, error) {
|
|
||||||
hostPort := net.JoinHostPort(p.IP.String(), strconv.Itoa(int(p.Port)))
|
|
||||||
conn, err := net.DialTimeout("tcp", hostPort, 3*time.Second)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
return conn, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (d *Download) handshake(conn net.Conn) (*handshake.Handshake, error) {
|
|
||||||
conn.SetDeadline(time.Now().Local().Add(3 * time.Second))
|
|
||||||
req := handshake.New(d.InfoHash, d.PeerID)
|
|
||||||
_, err := conn.Write(req.Serialize())
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
res, err := handshake.Read(conn)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
conn.SetDeadline(time.Time{}) // Disable the deadline
|
|
||||||
return res, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (d *Download) initPeer(p *Peer, c chan *peerState) {
|
|
||||||
conn, err := p.connect(d.PeerID, d.InfoHash)
|
|
||||||
if err != nil {
|
|
||||||
c <- nil
|
|
||||||
return
|
|
||||||
}
|
|
||||||
_, err = d.handshake(conn)
|
|
||||||
if err != nil {
|
|
||||||
c <- nil
|
|
||||||
return
|
|
||||||
}
|
|
||||||
c <- &peerState{p, conn}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (d *Download) startSwarm() *swarm {
|
|
||||||
c := make(chan *peerState)
|
|
||||||
for i := range d.Peers {
|
|
||||||
go d.initPeer(&d.Peers[i], c)
|
|
||||||
}
|
|
||||||
|
|
||||||
peerStates := make([]*peerState, 0)
|
|
||||||
for range d.Peers {
|
|
||||||
ps := <-c
|
|
||||||
if ps != nil {
|
|
||||||
peerStates = append(peerStates, ps)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return &swarm{peerStates}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Download downloads a torrent
|
// Download downloads a torrent
|
||||||
func (d *Download) Download() error {
|
func (d *Download) Download() error {
|
||||||
d.startSwarm()
|
clients := d.initClients()
|
||||||
|
if len(clients) == 0 {
|
||||||
|
return fmt.Errorf("Could not connect to any of %d clients", len(d.Peers))
|
||||||
|
}
|
||||||
|
|
||||||
|
queue := make(chan *pieceWork, len(d.PieceHashes))
|
||||||
|
for index, hash := range d.PieceHashes {
|
||||||
|
queue <- &pieceWork{index, hash}
|
||||||
|
}
|
||||||
|
processQueue(clients, queue)
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (d *Download) initClients() []*client {
|
||||||
|
// Create clients in parallel
|
||||||
|
c := make(chan *client)
|
||||||
|
for _, p := range d.Peers {
|
||||||
|
go func(p Peer) {
|
||||||
|
client, err := newClient(p, d.PeerID, d.InfoHash)
|
||||||
|
if err != nil {
|
||||||
|
c <- nil
|
||||||
|
} else {
|
||||||
|
c <- client
|
||||||
|
}
|
||||||
|
}(p)
|
||||||
|
}
|
||||||
|
|
||||||
|
clients := make([]*client, 0)
|
||||||
|
for range d.Peers {
|
||||||
|
client := <-c
|
||||||
|
if client != nil {
|
||||||
|
clients = append(clients, client)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return clients
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *swarm) selectClient(index int) *client {
|
||||||
|
return s.clients[0]
|
||||||
|
}
|
||||||
|
|
||||||
|
func processQueue(clients []*client, queue chan *pieceWork) {
|
||||||
|
s := swarm{clients, queue, sync.Mutex{}}
|
||||||
|
for pw := range s.queue {
|
||||||
|
client := s.selectClient(pw.index)
|
||||||
|
fmt.Println(client.conn.RemoteAddr())
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -6,6 +6,7 @@ import (
|
|||||||
"crypto/sha1"
|
"crypto/sha1"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
|
"net"
|
||||||
|
|
||||||
"github.com/jackpal/bencode-go"
|
"github.com/jackpal/bencode-go"
|
||||||
"github.com/veggiedefender/torrent-client/p2p"
|
"github.com/veggiedefender/torrent-client/p2p"
|
||||||
@@ -44,7 +45,8 @@ func (t *Torrent) Download() error {
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
peers, err := t.getPeers(peerID, Port)
|
// peers, err := t.getPeers(peerID, Port)
|
||||||
|
peers := []p2p.Peer{{IP: net.IP{127, 0, 0, 1}, Port: 51413}}
|
||||||
downloader := p2p.Download{
|
downloader := p2p.Download{
|
||||||
Peers: peers,
|
Peers: peers,
|
||||||
PeerID: peerID,
|
PeerID: peerID,
|
||||||
|
|||||||
Reference in New Issue
Block a user