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 }