+++ /dev/null
-package upnp
-
-import (
- "errors"
- "fmt"
- "net"
- "time"
-
- log "github.com/sirupsen/logrus"
-)
-
-type UPNPCapabilities struct {
- PortMapping bool
- Hairpin bool
-}
-
-func makeUPNPListener(intPort int, extPort int) (NAT, net.Listener, net.IP, error) {
- nat, err := Discover()
- if err != nil {
- return nil, nil, nil, errors.New(fmt.Sprintf("NAT upnp could not be discovered: %v", err))
- }
- log.WithField("ourIP", nat.(*upnpNAT).ourIP).Info("outIP:")
-
- ext, err := nat.GetExternalAddress()
- if err != nil {
- return nat, nil, nil, errors.New(fmt.Sprintf("External address error: %v", err))
- }
- log.WithField("address", ext).Info("External address")
-
- port, err := nat.AddPortMapping("tcp", extPort, intPort, "Tendermint UPnP Probe", 0)
- if err != nil {
- return nat, nil, ext, errors.New(fmt.Sprintf("Port mapping error: %v", err))
- }
- log.WithField("port", port).Info("Port mapping mapped")
-
- // also run the listener, open for all remote addresses.
- listener, err := net.Listen("tcp", fmt.Sprintf(":%v", intPort))
- if err != nil {
- return nat, nil, ext, errors.New(fmt.Sprintf("Error establishing listener: %v", err))
- }
- return nat, listener, ext, nil
-}
-
-func testHairpin(listener net.Listener, extAddr string) (supportsHairpin bool) {
- // Listener
- go func() {
- inConn, err := listener.Accept()
- if err != nil {
- log.WithField("error", err).Error("Listener.Accept() error")
- return
- }
- log.WithFields(log.Fields{
- "LocalAddr": inConn.LocalAddr(),
- "RemoteAddr": inConn.RemoteAddr(),
- }).Info("Accepted incoming connection")
- buf := make([]byte, 1024)
- n, err := inConn.Read(buf)
- if err != nil {
- log.WithField("error", err).Error("Incoming connection read error")
- return
- }
- log.Infof("Incoming connection read %v bytes: %X", n, buf)
- if string(buf) == "test data" {
- supportsHairpin = true
- return
- }
- }()
-
- // Establish outgoing
- outConn, err := net.Dial("tcp", extAddr)
- if err != nil {
- log.WithField("error", err).Error("Outgoing connection dial error")
- return
- }
-
- n, err := outConn.Write([]byte("test data"))
- if err != nil {
- log.WithField("error", err).Error("Outgoing connection write error")
- return
- }
- log.Infof("Outgoing connection wrote %v bytes", n)
-
- // Wait for data receipt
- time.Sleep(1 * time.Second)
- return
-}
-
-func Probe() (caps UPNPCapabilities, err error) {
- log.Info("Probing for UPnP!")
-
- intPort, extPort := 8001, 8001
-
- nat, listener, ext, err := makeUPNPListener(intPort, extPort)
- if err != nil {
- return
- }
- caps.PortMapping = true
-
- // Deferred cleanup
- defer func() {
- err = nat.DeletePortMapping("tcp", intPort, extPort)
- if err != nil {
- log.WithField("error", err).Error("Port mapping delete error")
- }
- listener.Close()
- }()
-
- supportsHairpin := testHairpin(listener, fmt.Sprintf("%v:%v", ext, extPort))
- if supportsHairpin {
- caps.Hairpin = true
- }
-
- return
-}