2017-08-08 13:58:51 +02:00
|
|
|
|
package mainline
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"net"
|
|
|
|
|
|
|
|
|
|
"github.com/anacrolix/torrent/bencode"
|
2018-07-24 14:41:13 +02:00
|
|
|
|
sockaddr "github.com/libp2p/go-sockaddr/net"
|
2017-11-03 00:15:13 +01:00
|
|
|
|
"go.uber.org/zap"
|
2018-04-25 22:33:50 +02:00
|
|
|
|
"golang.org/x/sys/unix"
|
2017-08-08 13:58:51 +02:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
type Transport struct {
|
2018-04-25 22:33:50 +02:00
|
|
|
|
fd int
|
2017-10-04 15:07:48 +02:00
|
|
|
|
laddr *net.UDPAddr
|
2017-08-08 13:58:51 +02:00
|
|
|
|
started bool
|
2018-07-24 14:41:13 +02:00
|
|
|
|
buffer []byte
|
2017-08-08 13:58:51 +02:00
|
|
|
|
|
|
|
|
|
// OnMessage is the function that will be called when Transport receives a packet that is
|
|
|
|
|
// successfully unmarshalled as a syntactically correct Message (but -of course- the checking
|
|
|
|
|
// the semantic correctness of the Message is left to Protocol).
|
|
|
|
|
onMessage func(*Message, net.Addr)
|
2018-07-24 14:41:13 +02:00
|
|
|
|
// OnCongestion
|
|
|
|
|
onCongestion func()
|
2017-08-08 13:58:51 +02:00
|
|
|
|
}
|
|
|
|
|
|
2018-07-24 14:41:13 +02:00
|
|
|
|
func NewTransport(laddr string, onMessage func(*Message, net.Addr), onCongestion func()) *Transport {
|
|
|
|
|
t := new(Transport)
|
|
|
|
|
/* The field size sets a theoretical limit of 65,535 bytes (8 byte header + 65,527 bytes of
|
|
|
|
|
* data) for a UDP datagram. However the actual limit for the data length, which is imposed by
|
|
|
|
|
* the underlying IPv4 protocol, is 65,507 bytes (65,535 − 8 byte UDP header − 20 byte IP
|
|
|
|
|
* header).
|
|
|
|
|
*
|
|
|
|
|
* In IPv6 jumbograms it is possible to have UDP packets of size greater than 65,535 bytes.
|
|
|
|
|
* RFC 2675 specifies that the length field is set to zero if the length of the UDP header plus
|
|
|
|
|
* UDP data is greater than 65,535.
|
|
|
|
|
*
|
|
|
|
|
* https://en.wikipedia.org/wiki/User_Datagram_Protocol
|
|
|
|
|
*/
|
|
|
|
|
t.buffer = make([]byte, 65507)
|
|
|
|
|
t.onMessage = onMessage
|
|
|
|
|
t.onCongestion = onCongestion
|
|
|
|
|
|
2017-11-03 00:15:13 +01:00
|
|
|
|
var err error
|
2018-07-24 14:41:13 +02:00
|
|
|
|
t.laddr, err = net.ResolveUDPAddr("udp", laddr)
|
2017-10-04 15:07:48 +02:00
|
|
|
|
if err != nil {
|
|
|
|
|
zap.L().Panic("Could not resolve the UDP address for the trawler!", zap.Error(err))
|
|
|
|
|
}
|
2018-07-24 14:41:13 +02:00
|
|
|
|
if t.laddr.IP.To4() == nil {
|
|
|
|
|
zap.L().Panic("IP address is not IPv4!")
|
|
|
|
|
}
|
2017-08-08 13:58:51 +02:00
|
|
|
|
|
2018-07-24 14:41:13 +02:00
|
|
|
|
return t
|
2017-08-08 13:58:51 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (t *Transport) Start() {
|
|
|
|
|
// Why check whether the Transport `t` started or not, here and not -for instance- in
|
|
|
|
|
// t.Terminate()?
|
|
|
|
|
// Because in t.Terminate() the programmer (i.e. you & me) would stumble upon an error while
|
|
|
|
|
// trying close an uninitialised net.UDPConn or something like that: it's mostly harmless
|
|
|
|
|
// because its effects are immediate. But if you try to start a Transport `t` for the second
|
|
|
|
|
// (or the third, 4th, ...) time, it will keep spawning goroutines and any small mistake may
|
|
|
|
|
// end up in a debugging horror.
|
|
|
|
|
// Here ends my justification.
|
|
|
|
|
if t.started {
|
|
|
|
|
zap.L().Panic("Attempting to Start() a mainline/Transport that has been already started! (Programmer error.)")
|
|
|
|
|
}
|
|
|
|
|
t.started = true
|
|
|
|
|
|
|
|
|
|
var err error
|
2018-04-25 22:33:50 +02:00
|
|
|
|
t.fd, err = unix.Socket(unix.SOCK_DGRAM, unix.AF_INET, 0)
|
2017-08-08 13:58:51 +02:00
|
|
|
|
if err != nil {
|
|
|
|
|
zap.L().Fatal("Could NOT create a UDP socket!", zap.Error(err))
|
|
|
|
|
}
|
|
|
|
|
|
2018-07-24 14:41:13 +02:00
|
|
|
|
var ip [4]byte
|
|
|
|
|
copy(ip[:], t.laddr.IP.To4())
|
|
|
|
|
err = unix.Bind(t.fd, &unix.SockaddrInet4{Addr: ip, Port: t.laddr.Port})
|
|
|
|
|
if err != nil {
|
|
|
|
|
zap.L().Fatal("Could NOT bind the socket!", zap.Error(err))
|
|
|
|
|
}
|
2018-04-25 22:33:50 +02:00
|
|
|
|
|
2017-08-08 13:58:51 +02:00
|
|
|
|
go t.readMessages()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (t *Transport) Terminate() {
|
2018-04-25 22:33:50 +02:00
|
|
|
|
unix.Close(t.fd);
|
2017-08-08 13:58:51 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// readMessages is a goroutine!
|
|
|
|
|
func (t *Transport) readMessages() {
|
|
|
|
|
for {
|
2018-07-24 14:41:13 +02:00
|
|
|
|
n, fromSA, err := unix.Recvfrom(t.fd, t.buffer, 0)
|
|
|
|
|
if err == unix.EPERM || err == unix.ENOBUFS { // todo: are these errors possible for recvfrom?
|
|
|
|
|
zap.L().Warn("READ CONGESTION!", zap.Error(err))
|
|
|
|
|
t.onCongestion()
|
|
|
|
|
} else if err != nil {
|
2017-08-08 13:58:51 +02:00
|
|
|
|
// TODO: isn't there a more reliable way to detect if UDPConn is closed?
|
2018-04-25 22:33:50 +02:00
|
|
|
|
zap.L().Debug("Could NOT read an UDP packet!", zap.Error(err))
|
2017-08-08 13:58:51 +02:00
|
|
|
|
}
|
|
|
|
|
|
2018-07-24 14:41:13 +02:00
|
|
|
|
if n == 0 {
|
|
|
|
|
/* Datagram sockets in various domains (e.g., the UNIX and Internet domains) permit
|
|
|
|
|
* zero-length datagrams. When such a datagram is received, the return value (n) is 0.
|
|
|
|
|
*/
|
|
|
|
|
zap.L().Debug("zero-length received!!")
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
from := sockaddr.SockaddrToUDPAddr(fromSA)
|
|
|
|
|
|
2017-08-08 13:58:51 +02:00
|
|
|
|
var msg Message
|
2018-07-24 14:41:13 +02:00
|
|
|
|
err = bencode.Unmarshal(t.buffer[:n], &msg)
|
2017-08-08 13:58:51 +02:00
|
|
|
|
if err != nil {
|
|
|
|
|
zap.L().Debug("Could NOT unmarshal packet data!", zap.Error(err))
|
|
|
|
|
}
|
|
|
|
|
|
2018-07-24 14:41:13 +02:00
|
|
|
|
zap.L().Debug("message read! (first 20...)", zap.ByteString("msg", t.buffer[:20]))
|
2018-04-25 22:33:50 +02:00
|
|
|
|
t.onMessage(&msg, from)
|
2017-08-08 13:58:51 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (t *Transport) WriteMessages(msg *Message, addr net.Addr) {
|
|
|
|
|
data, err := bencode.Marshal(msg)
|
|
|
|
|
if err != nil {
|
|
|
|
|
zap.L().Panic("Could NOT marshal an outgoing message! (Programmer error.)")
|
|
|
|
|
}
|
|
|
|
|
|
2018-07-24 14:41:13 +02:00
|
|
|
|
addrSA := sockaddr.NetAddrToSockaddr(addr)
|
|
|
|
|
|
|
|
|
|
zap.L().Debug("sent message!!!")
|
|
|
|
|
|
|
|
|
|
err = unix.Sendto(t.fd, data, 0, addrSA)
|
|
|
|
|
if err == unix.EPERM || err == unix.ENOBUFS {
|
|
|
|
|
/* EPERM (errno: 1) is kernel's way of saying that "you are far too fast, chill". It is
|
|
|
|
|
* also likely that we have received a ICMP source quench packet (meaning, that we *really*
|
|
|
|
|
* need to slow down.
|
|
|
|
|
*
|
|
|
|
|
* Read more here: http://www.archivum.info/comp.protocols.tcp-ip/2009-05/00088/UDP-socket-amp-amp-sendto-amp-amp-EPERM.html
|
|
|
|
|
*
|
|
|
|
|
* > Note On BSD systems (OS X, FreeBSD, etc.) flow control is not supported for
|
|
|
|
|
* > DatagramProtocol, because send failures caused by writing too many packets cannot be
|
|
|
|
|
* > detected easily. The socket always appears ‘ready’ and excess packets are dropped; an
|
|
|
|
|
* > OSError with errno set to errno.ENOBUFS may or may not be raised; if it is raised, it
|
|
|
|
|
* > will be reported to DatagramProtocol.error_received() but otherwise ignored.
|
|
|
|
|
*
|
|
|
|
|
* Source: https://docs.python.org/3/library/asyncio-protocol.html#flow-control-callbacks
|
|
|
|
|
*/
|
|
|
|
|
zap.L().Warn("WRITE CONGESTION!", zap.Error(err))
|
|
|
|
|
t.onCongestion()
|
|
|
|
|
} else if err != nil {
|
2017-08-08 13:58:51 +02:00
|
|
|
|
zap.L().Debug("Could NOT write an UDP packet!", zap.Error(err))
|
|
|
|
|
}
|
|
|
|
|
}
|