Files
mochi/frontend/udp/writer.go
Lawrence, Rendall 61f859e3f6 (partially tested) simplify client approval m/w
* sanitize code
2023-03-23 00:34:10 +03:00

92 lines
2.7 KiB
Go

package udp
import (
"encoding/binary"
"errors"
"io"
"time"
"github.com/sot-tech/mochi/bittorrent"
)
// writeErrorResponse writes the failure reason as a null-terminated string.
func writeErrorResponse(w io.Writer, txID []byte, err error) {
buf := reqRespBufferPool.Get()
defer reqRespBufferPool.Put(buf)
writeHeader(buf, txID, errorActionID)
message := "mochi internal error"
var clientErr bittorrent.ClientError
// If the client wasn't at fault, acknowledge it.
if errors.As(err, &clientErr) {
message = clientErr.Error()
} else {
logger.Error().Err(err).Msg("internal error")
}
buf.WriteString(message)
buf.WriteByte('\000')
_, _ = buf.WriteTo(w)
}
// writeAnnounceResponse encodes an announce response according to BEP 15.
// The peers returned will be resp.IPv6Peers or resp.IPv4Peers, depending on
// whether v6Peers is set.
// If v6Action is set, the action will be 4, according to
// https://web.archive.org/web/20170503181830/http://opentracker.blog.h3q.com/2007/12/28/the-ipv6-situation/
func writeAnnounceResponse(w io.Writer, txID []byte, resp *bittorrent.AnnounceResponse, v6Action, v6Peers bool) {
buf := reqRespBufferPool.Get()
defer reqRespBufferPool.Put(buf)
if v6Action {
writeHeader(buf, txID, announceV6ActionID)
} else {
writeHeader(buf, txID, announceActionID)
}
_ = binary.Write(buf, binary.BigEndian, uint32(resp.Interval/time.Second))
_ = binary.Write(buf, binary.BigEndian, resp.Incomplete)
_ = binary.Write(buf, binary.BigEndian, resp.Complete)
peers := resp.IPv4Peers
if v6Peers {
peers = resp.IPv6Peers
}
for _, peer := range peers {
buf.Write(peer.Addr().AsSlice())
_ = binary.Write(buf, binary.BigEndian, peer.Port())
}
_, _ = buf.WriteTo(w)
}
// writeScrapeResponse encodes a scrape response according to BEP 15.
func writeScrapeResponse(w io.Writer, txID []byte, resp *bittorrent.ScrapeResponse) {
buf := reqRespBufferPool.Get()
defer reqRespBufferPool.Put(buf)
writeHeader(buf, txID, scrapeActionID)
for _, scrape := range resp.Data {
_ = binary.Write(buf, binary.BigEndian, scrape.Complete)
_ = binary.Write(buf, binary.BigEndian, scrape.Snatches)
_ = binary.Write(buf, binary.BigEndian, scrape.Incomplete)
}
_, _ = buf.WriteTo(w)
}
// writeConnectionID encodes a new connection response according to BEP 15.
func writeConnectionID(w io.Writer, txID, connID []byte) {
buf := reqRespBufferPool.Get()
defer reqRespBufferPool.Put(buf)
writeHeader(buf, txID, connectActionID)
buf.Write(connID)
_, _ = buf.WriteTo(w)
}
// writeHeader writes the action and transaction ID to the provided response
// buffer.
func writeHeader(w io.Writer, txID []byte, action uint32) {
_ = binary.Write(w, binary.BigEndian, action)
_, _ = w.Write(txID)
}