tendermint/p2p/listener.go

287 lines
7.2 KiB
Go
Raw Normal View History

2015-10-25 18:21:51 -07:00
package p2p
import (
"fmt"
"net"
"strconv"
2018-06-28 00:29:27 -07:00
"strings"
2015-10-25 18:21:51 -07:00
"time"
2018-07-01 19:36:49 -07:00
cmn "github.com/tendermint/tendermint/libs/common"
"github.com/tendermint/tendermint/libs/log"
"github.com/tendermint/tendermint/p2p/upnp"
2015-10-25 18:21:51 -07:00
)
// Listener is a network listener for stream-oriented protocols, providing
// convenient methods to get listener's internal and external addresses.
// Clients are supposed to read incoming connections from a channel, returned
// by Connections() method.
2015-10-25 18:21:51 -07:00
type Listener interface {
Connections() <-chan net.Conn
InternalAddress() *NetAddress
ExternalAddress() *NetAddress
ExternalAddressHost() string
2015-10-25 18:21:51 -07:00
String() string
Stop() error
2015-10-25 18:21:51 -07:00
}
// DefaultListener is a cmn.Service, running net.Listener underneath.
// Optionally, UPnP is used upon calling NewDefaultListener to resolve external
// address.
2015-10-25 18:21:51 -07:00
type DefaultListener struct {
2017-05-01 19:12:14 -07:00
cmn.BaseService
2015-10-25 18:21:51 -07:00
listener net.Listener
intAddr *NetAddress
extAddr *NetAddress
connections chan net.Conn
}
var _ Listener = (*DefaultListener)(nil)
2015-10-25 18:21:51 -07:00
const (
numBufferedConnections = 10
defaultExternalPort = 8770
tryListenSeconds = 5
)
func splitHostPort(addr string) (host string, port int) {
host, portStr, err := net.SplitHostPort(addr)
if err != nil {
2017-12-15 11:35:49 -08:00
panic(err)
2015-10-25 18:21:51 -07:00
}
port, err = strconv.Atoi(portStr)
if err != nil {
2017-12-15 11:35:49 -08:00
panic(err)
2015-10-25 18:21:51 -07:00
}
return host, port
}
// NewDefaultListener creates a new DefaultListener on lAddr, optionally trying
// to determine external address using UPnP.
2018-07-02 09:18:17 -07:00
func NewDefaultListener(
fullListenAddrString string,
externalAddrString string,
useUPnP bool,
logger log.Logger) Listener {
// Split protocol, address, and port.
2018-07-02 09:18:17 -07:00
protocol, lAddr := cmn.ProtocolAndAddress(fullListenAddrString)
2015-10-25 18:21:51 -07:00
lAddrIP, lAddrPort := splitHostPort(lAddr)
// Create listener
var listener net.Listener
var err error
for i := 0; i < tryListenSeconds; i++ {
listener, err = net.Listen(protocol, lAddr)
if err == nil {
break
} else if i < tryListenSeconds-1 {
time.Sleep(time.Second * 1)
}
}
if err != nil {
2017-12-15 11:35:49 -08:00
panic(err)
2015-10-25 18:21:51 -07:00
}
// Actual listener local IP & port
listenerIP, listenerPort := splitHostPort(listener.Addr().String())
2017-05-02 00:53:32 -07:00
logger.Info("Local listener", "ip", listenerIP, "port", listenerPort)
2015-10-25 18:21:51 -07:00
// Determine internal address...
2017-03-02 19:00:50 -08:00
var intAddr *NetAddress
intAddr, err = NewNetAddressStringWithOptionalID(lAddr)
2017-03-02 19:00:50 -08:00
if err != nil {
2017-12-15 11:35:49 -08:00
panic(err)
2017-03-02 19:00:50 -08:00
}
2015-10-25 18:21:51 -07:00
inAddrAny := lAddrIP == "" || lAddrIP == "0.0.0.0"
2018-07-02 09:18:17 -07:00
// Determine external address.
2015-10-25 18:21:51 -07:00
var extAddr *NetAddress
2018-07-02 09:18:17 -07:00
if externalAddrString != "" {
var err error
2018-07-02 09:18:17 -07:00
extAddr, err = NewNetAddressStringWithOptionalID(externalAddrString)
if err != nil {
panic(fmt.Sprintf("Error in ExternalAddress: %v", err))
}
}
2018-07-02 09:18:17 -07:00
// If the lAddrIP is INADDR_ANY, try UPnP.
if extAddr == nil && useUPnP && inAddrAny {
extAddr = getUPNPExternalAddress(lAddrPort, listenerPort, logger)
}
// Otherwise just use the local address.
2015-10-25 18:21:51 -07:00
if extAddr == nil {
defaultToIPv4 := inAddrAny
extAddr = getNaiveExternalAddress(defaultToIPv4, listenerPort, false, logger)
2015-10-25 18:21:51 -07:00
}
if extAddr == nil {
2017-12-15 11:35:49 -08:00
panic("Could not determine external address!")
2015-10-25 18:21:51 -07:00
}
dl := &DefaultListener{
listener: listener,
intAddr: intAddr,
extAddr: extAddr,
connections: make(chan net.Conn, numBufferedConnections),
}
2017-05-02 00:53:32 -07:00
dl.BaseService = *cmn.NewBaseService(logger, "DefaultListener", dl)
err = dl.Start() // Started upon construction
2017-09-21 09:38:48 -07:00
if err != nil {
logger.Error("Error starting base service", "err", err)
}
2015-10-25 18:21:51 -07:00
return dl
}
// OnStart implements cmn.Service by spinning a goroutine, listening for new
// connections.
2015-10-25 18:21:51 -07:00
func (l *DefaultListener) OnStart() error {
2017-09-21 09:38:48 -07:00
if err := l.BaseService.OnStart(); err != nil {
return err
}
2015-10-25 18:21:51 -07:00
go l.listenRoutine()
return nil
}
// OnStop implements cmn.Service by closing the listener.
2015-10-25 18:21:51 -07:00
func (l *DefaultListener) OnStop() {
l.BaseService.OnStop()
2017-10-03 16:48:43 -07:00
l.listener.Close() // nolint: errcheck
2015-10-25 18:21:51 -07:00
}
2018-07-24 18:32:30 -07:00
// Accept connections and pass on the channel.
2015-10-25 18:21:51 -07:00
func (l *DefaultListener) listenRoutine() {
for {
conn, err := l.listener.Accept()
if !l.IsRunning() {
break // Go to cleanup
}
// listener wasn't stopped,
// yet we encountered an error.
if err != nil {
2017-12-15 11:35:49 -08:00
panic(err)
2015-10-25 18:21:51 -07:00
}
l.connections <- conn
}
// Cleanup
close(l.connections)
for range l.connections {
2015-10-25 18:21:51 -07:00
// Drain
}
}
// Connections returns a channel of inbound connections.
2015-10-25 18:21:51 -07:00
// It gets closed when the listener closes.
2018-07-24 18:32:30 -07:00
// It is the callers responsibility to close any connections received
// over this channel.
2015-10-25 18:21:51 -07:00
func (l *DefaultListener) Connections() <-chan net.Conn {
return l.connections
}
// InternalAddress returns the internal NetAddress (address used for
// listening).
2015-10-25 18:21:51 -07:00
func (l *DefaultListener) InternalAddress() *NetAddress {
return l.intAddr
}
// ExternalAddress returns the external NetAddress (publicly available,
// determined using either UPnP or local resolver).
2015-10-25 18:21:51 -07:00
func (l *DefaultListener) ExternalAddress() *NetAddress {
return l.extAddr
}
// ExternalAddressHost returns the external NetAddress IP string. If an IP is
// IPv6, it's wrapped in brackets ("[2001:db8:1f70::999:de8:7648:6e8]").
func (l *DefaultListener) ExternalAddressHost() string {
2018-06-28 00:29:27 -07:00
ip := l.ExternalAddress().IP
if isIpv6(ip) {
// Means it's ipv6, so format it with brackets
return "[" + ip.String() + "]"
}
return ip.String()
}
2015-10-25 18:21:51 -07:00
func (l *DefaultListener) String() string {
return fmt.Sprintf("Listener(@%v)", l.extAddr)
}
/* external address helpers */
// UPNP external address discovery & port mapping
2017-05-02 00:53:32 -07:00
func getUPNPExternalAddress(externalPort, internalPort int, logger log.Logger) *NetAddress {
logger.Info("Getting UPNP external address")
2015-10-25 18:21:51 -07:00
nat, err := upnp.Discover()
if err != nil {
logger.Info("Could not perform UPNP discover", "err", err)
2015-10-25 18:21:51 -07:00
return nil
}
ext, err := nat.GetExternalAddress()
if err != nil {
logger.Info("Could not get UPNP external address", "err", err)
2015-10-25 18:21:51 -07:00
return nil
}
// UPnP can't seem to get the external port, so let's just be explicit.
if externalPort == 0 {
externalPort = defaultExternalPort
}
externalPort, err = nat.AddPortMapping("tcp", externalPort, internalPort, "tendermint", 0)
if err != nil {
logger.Info("Could not add UPNP port mapping", "err", err)
2015-10-25 18:21:51 -07:00
return nil
}
2017-05-02 00:53:32 -07:00
logger.Info("Got UPNP external address", "address", ext)
2015-10-25 18:21:51 -07:00
return NewNetAddressIPPort(ext, uint16(externalPort))
}
2018-06-28 00:29:27 -07:00
func isIpv6(ip net.IP) bool {
v4 := ip.To4()
if v4 != nil {
return false
}
ipString := ip.String()
// Extra check just to be sure it's IPv6
return (strings.Contains(ipString, ":") && !strings.Contains(ipString, "."))
}
// TODO: use syscalls: see issue #712
func getNaiveExternalAddress(defaultToIPv4 bool, port int, settleForLocal bool, logger log.Logger) *NetAddress {
2015-10-25 18:21:51 -07:00
addrs, err := net.InterfaceAddrs()
if err != nil {
2017-12-15 11:35:49 -08:00
panic(cmn.Fmt("Could not fetch interface addresses: %v", err))
2015-10-25 18:21:51 -07:00
}
for _, a := range addrs {
ipnet, ok := a.(*net.IPNet)
if !ok {
continue
}
if defaultToIPv4 || !isIpv6(ipnet.IP) {
2018-06-28 00:29:27 -07:00
v4 := ipnet.IP.To4()
if v4 == nil || (!settleForLocal && v4[0] == 127) {
// loopback
continue
}
} else if !settleForLocal && ipnet.IP.IsLoopback() {
// IPv6, check for loopback
2015-10-25 18:21:51 -07:00
continue
2018-06-28 00:29:27 -07:00
}
2015-10-25 18:21:51 -07:00
return NewNetAddressIPPort(ipnet.IP, uint16(port))
}
// try again, but settle for local
logger.Info("Node may not be connected to internet. Settling for local address")
return getNaiveExternalAddress(defaultToIPv4, port, true, logger)
2015-10-25 18:21:51 -07:00
}