package p2p
|
|
|
|
import (
|
|
"net"
|
|
"strconv"
|
|
"sync/atomic"
|
|
|
|
. "github.com/tendermint/tendermint/binary"
|
|
. "github.com/tendermint/tendermint/common"
|
|
"github.com/tendermint/tendermint/peer/upnp"
|
|
)
|
|
|
|
/*
|
|
Listener is part of a Server.
|
|
*/
|
|
type Listener interface {
|
|
Connections() <-chan *Connection
|
|
ExternalAddress() *NetAddress
|
|
Stop()
|
|
}
|
|
|
|
/*
|
|
DefaultListener is an implementation that works on the golang network stack.
|
|
*/
|
|
type DefaultListener struct {
|
|
listener net.Listener
|
|
extAddr *NetAddress
|
|
connections chan *Connection
|
|
stopped uint32
|
|
}
|
|
|
|
const (
|
|
NumBufferedConnections = 10
|
|
)
|
|
|
|
func NewDefaultListener(protocol string, listenAddr string) Listener {
|
|
listenHost, listenPortStr, err := net.SplitHostPort(listenAddr)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
listenPort, err := strconv.Atoi(listenPortStr)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
// Determine external address...
|
|
var extAddr *NetAddress
|
|
// If the listenHost is INADDR_ANY, try UPnP
|
|
if listenHost == "" || listenHost == "0.0.0.0" {
|
|
extAddr = getUPNPExternalAddress(listenPort, listenPort)
|
|
}
|
|
// Otherwise just use the local address...
|
|
if extAddr == nil {
|
|
extAddr = getNaiveExternalAddress(listenPort)
|
|
}
|
|
if extAddr == nil {
|
|
panic("Could not determine external address!")
|
|
}
|
|
|
|
// Create listener
|
|
listener, err := net.Listen(protocol, listenAddr)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
dl := &DefaultListener{
|
|
listener: listener,
|
|
extAddr: extAddr,
|
|
connections: make(chan *Connection, NumBufferedConnections),
|
|
}
|
|
|
|
go dl.listenHandler()
|
|
|
|
return dl
|
|
}
|
|
|
|
func (l *DefaultListener) listenHandler() {
|
|
for {
|
|
conn, err := l.listener.Accept()
|
|
|
|
if atomic.LoadUint32(&l.stopped) == 1 {
|
|
break // go to cleanup
|
|
}
|
|
|
|
// listener wasn't stopped,
|
|
// yet we encountered an error.
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
c := NewConnection(conn)
|
|
l.connections <- c
|
|
}
|
|
|
|
// cleanup
|
|
close(l.connections)
|
|
for _ = range l.connections {
|
|
// drain
|
|
}
|
|
}
|
|
|
|
func (l *DefaultListener) Connections() <-chan *Connection {
|
|
return l.connections
|
|
}
|
|
|
|
func (l *DefaultListener) ExternalAddress() *NetAddress {
|
|
return l.extAddr
|
|
}
|
|
|
|
func (l *DefaultListener) Stop() {
|
|
if atomic.CompareAndSwapUint32(&l.stopped, 0, 1) {
|
|
l.listener.Close()
|
|
}
|
|
}
|
|
|
|
/* external address helpers */
|
|
|
|
// UPNP external address discovery & port mapping
|
|
func getUPNPExternalAddress(externalPort, internalPort int) *NetAddress {
|
|
log.Infof("Getting UPNP external address")
|
|
nat, err := upnp.Discover()
|
|
if err != nil {
|
|
log.Infof("Could not get UPNP extrernal address: %v", err)
|
|
return nil
|
|
}
|
|
|
|
ext, err := nat.GetExternalAddress()
|
|
if err != nil {
|
|
log.Infof("Could not get UPNP external address: %v", err)
|
|
return nil
|
|
}
|
|
|
|
externalPort, err = nat.AddPortMapping("tcp", externalPort, internalPort, "tendermint", 0)
|
|
if err != nil {
|
|
log.Infof("Could not get UPNP external address: %v", err)
|
|
return nil
|
|
}
|
|
|
|
log.Infof("Got UPNP external address: %v", ext)
|
|
return NewNetAddressIPPort(ext, UInt16(externalPort))
|
|
}
|
|
|
|
// TODO: use syscalls: http://pastebin.com/9exZG4rh
|
|
func getNaiveExternalAddress(port int) *NetAddress {
|
|
addrs, err := net.InterfaceAddrs()
|
|
if err != nil {
|
|
Panicf("Unexpected error fetching interface addresses: %v", err)
|
|
}
|
|
|
|
for _, a := range addrs {
|
|
ipnet, ok := a.(*net.IPNet)
|
|
if !ok {
|
|
continue
|
|
}
|
|
v4 := ipnet.IP.To4()
|
|
if v4 == nil || v4[0] == 127 {
|
|
continue
|
|
} // loopback
|
|
return NewNetAddressIPPort(ipnet.IP, UInt16(port))
|
|
}
|
|
return nil
|
|
}
|