2016-08-03 09:11:52 +02:00
|
|
|
package udp
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"encoding/binary"
|
|
|
|
"fmt"
|
2016-08-05 07:47:04 +02:00
|
|
|
"io"
|
2016-08-03 09:11:52 +02:00
|
|
|
"time"
|
|
|
|
|
2016-08-17 03:42:08 +02:00
|
|
|
"github.com/chihaya/chihaya/bittorrent"
|
2016-08-03 09:11:52 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
// WriteError writes the failure reason as a null-terminated string.
|
2016-08-05 07:47:04 +02:00
|
|
|
func WriteError(w io.Writer, txID []byte, err error) {
|
2016-08-03 09:11:52 +02:00
|
|
|
// If the client wasn't at fault, acknowledge it.
|
|
|
|
if _, ok := err.(bittorrent.ClientError); !ok {
|
|
|
|
err = fmt.Errorf("internal error occurred: %s", err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
var buf bytes.Buffer
|
2016-08-05 07:47:04 +02:00
|
|
|
writeHeader(&buf, txID, errorActionID)
|
2016-08-03 09:11:52 +02:00
|
|
|
buf.WriteString(err.Error())
|
|
|
|
buf.WriteRune('\000')
|
2016-08-05 07:47:04 +02:00
|
|
|
w.Write(buf.Bytes())
|
2016-08-03 09:11:52 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// WriteAnnounce encodes an announce response according to BEP 15.
|
2016-08-05 07:47:04 +02:00
|
|
|
func WriteAnnounce(w io.Writer, txID []byte, resp *bittorrent.AnnounceResponse) {
|
2016-08-18 01:04:26 +02:00
|
|
|
var buf bytes.Buffer
|
|
|
|
|
|
|
|
writeHeader(&buf, txID, announceActionID)
|
|
|
|
binary.Write(&buf, binary.BigEndian, uint32(resp.Interval/time.Second))
|
|
|
|
binary.Write(&buf, binary.BigEndian, uint32(resp.Incomplete))
|
|
|
|
binary.Write(&buf, binary.BigEndian, uint32(resp.Complete))
|
2016-08-03 09:11:52 +02:00
|
|
|
|
|
|
|
for _, peer := range resp.IPv4Peers {
|
2016-08-18 01:04:26 +02:00
|
|
|
buf.Write(peer.IP)
|
|
|
|
binary.Write(&buf, binary.BigEndian, peer.Port)
|
2016-08-03 09:11:52 +02:00
|
|
|
}
|
2016-08-18 01:04:26 +02:00
|
|
|
|
|
|
|
w.Write(buf.Bytes())
|
2016-08-03 09:11:52 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// WriteScrape encodes a scrape response according to BEP 15.
|
2016-08-05 07:47:04 +02:00
|
|
|
func WriteScrape(w io.Writer, txID []byte, resp *bittorrent.ScrapeResponse) {
|
2016-08-18 01:04:26 +02:00
|
|
|
var buf bytes.Buffer
|
|
|
|
|
|
|
|
writeHeader(&buf, txID, scrapeActionID)
|
2016-08-03 09:11:52 +02:00
|
|
|
|
|
|
|
for _, scrape := range resp.Files {
|
2016-08-18 01:04:26 +02:00
|
|
|
binary.Write(&buf, binary.BigEndian, scrape.Complete)
|
|
|
|
binary.Write(&buf, binary.BigEndian, scrape.Snatches)
|
|
|
|
binary.Write(&buf, binary.BigEndian, scrape.Incomplete)
|
2016-08-03 09:11:52 +02:00
|
|
|
}
|
2016-08-18 01:04:26 +02:00
|
|
|
|
|
|
|
w.Write(buf.Bytes())
|
2016-08-03 09:11:52 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// WriteConnectionID encodes a new connection response according to BEP 15.
|
2016-08-05 07:47:04 +02:00
|
|
|
func WriteConnectionID(w io.Writer, txID, connID []byte) {
|
2016-08-18 01:04:26 +02:00
|
|
|
var buf bytes.Buffer
|
|
|
|
|
|
|
|
writeHeader(&buf, txID, connectActionID)
|
|
|
|
buf.Write(connID)
|
|
|
|
|
|
|
|
w.Write(buf.Bytes())
|
2016-08-03 09:11:52 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// writeHeader writes the action and transaction ID to the provided response
|
|
|
|
// buffer.
|
2016-08-05 07:47:04 +02:00
|
|
|
func writeHeader(w io.Writer, txID []byte, action uint32) {
|
|
|
|
binary.Write(w, binary.BigEndian, action)
|
|
|
|
w.Write(txID)
|
2016-08-03 09:11:52 +02:00
|
|
|
}
|