Files
go-upnp/upnp.go
2021-10-02 19:35:47 +09:00

228 lines
7.3 KiB
Go

// Package upnp provides a simple and opinionated interface to UPnP-enabled
// routers, allowing users to forward ports and discover their external IP
// address. Specific quirks:
//
// - When attempting to discover UPnP-enabled routers on the network, only the
// first such router is returned. If you have multiple routers, this may cause
// some trouble. But why would you do that?
//
// - Forwarded ports are always symmetric, e.g. the router's port 9980 will be
// mapped to the client's port 9980. This will be unacceptable for some
// purposes, but too bad. Symmetric mappings are the desired behavior 99% of
// the time, and they save a function argument.
//
// - TCP and UDP protocols are forwarded together.
//
// - Ports are forwarded permanently. Some other implementations lease a port
// mapping for a set duration, and then renew it periodically. This is nice,
// because it means mappings won't stick around after they've served their
// purpose. Unfortunately, some routers only support permanent mappings, so this
// package has chosen to support the lowest common denominator. To un-forward a
// port, you must use the Clear function (or do it manually).
//
// Once you've discovered your router, you can retrieve its address by calling
// its Location method. This address can be supplied to Load to connect to the
// router directly, which is much faster than calling Discover.
package upnp
import (
"context"
"errors"
"net"
"net/url"
"strings"
"time"
"gitlab.com/NebulousLabs/fastrand"
"gitlab.com/NebulousLabs/go-upnp/goupnp"
"gitlab.com/NebulousLabs/go-upnp/goupnp/dcps/internetgateway1"
)
// An IGD provides an interface to the most commonly used functions of an
// Internet Gateway Device: discovering the external IP, and forwarding ports.
type IGD struct {
// This interface is satisfied by the internetgateway1.WANIPConnection1
// and internetgateway1.WANPPPConnection1 types.
client interface {
GetExternalIPAddress() (string, error)
AddPortMapping(string, uint16, string, uint16, string, bool, string, uint32) error
GetSpecificPortMappingEntry(string, uint16, string) (uint16, string, bool, string, uint32, error)
DeletePortMapping(string, uint16, string) error
GetServiceClient() *goupnp.ServiceClient
}
}
// ExternalIP returns the router's external IP.
func (d *IGD) ExternalIP() (string, error) {
return d.client.GetExternalIPAddress()
}
// IsForwardedTCP checks whether a specific TCP port is forwarded to this host
func (d *IGD) IsForwardedTCP(port uint16) (bool, error) {
return d.checkForward(port, "TCP")
}
// IsForwardedUDP checks whether a specific UDP port is forwarded to this host
func (d *IGD) IsForwardedUDP(port uint16) (bool, error) {
return d.checkForward(port, "UDP")
}
// checkForward checks whether a specific TCP or UDP port is forwarded to this host
func (d *IGD) checkForward(port uint16, proto string) (bool, error) {
time.Sleep(time.Millisecond)
_, _, enabled, _, _, err := d.client.GetSpecificPortMappingEntry("", port, proto)
if err != nil {
// 714 "NoSuchEntryInArray" means that there is no such forwarding
if strings.Contains(err.Error(), "<errorCode>714</errorCode>") {
return false, nil
}
return false, err
}
return enabled, nil
}
// Forward forwards the specified port, and adds its description to the
// router's port mapping table.
func (d *IGD) Forward(port uint16, desc string) error {
ip, err := d.getInternalIP()
if err != nil {
return err
}
time.Sleep(time.Millisecond)
err = d.client.AddPortMapping("", port, "TCP", port, ip, true, desc, 0)
if err != nil {
return err
}
time.Sleep(time.Millisecond)
return d.client.AddPortMapping("", port, "UDP", port, ip, true, desc, 0)
}
// ForwardTCP forwards the specified TCP port, and adds its description to the
// router's port mapping table.
func (d *IGD) ForwardTCP(port uint16, desc string) error {
ip, err := d.getInternalIP()
if err != nil {
return err
}
time.Sleep(time.Millisecond)
return d.client.AddPortMapping("", port, "TCP", port, ip, true, desc, 0)
}
// ForwardUDP forwards the specified UDP port, and adds its description to the
// router's port mapping table.
func (d *IGD) ForwardUDP(port uint16, desc string) error {
ip, err := d.getInternalIP()
if err != nil {
return err
}
time.Sleep(time.Millisecond)
return d.client.AddPortMapping("", port, "UDP", port, ip, true, desc, 0)
}
// Clear un-forwards a port, removing it from the router's port mapping table.
func (d *IGD) Clear(port uint16) error {
time.Sleep(time.Millisecond)
tcpErr := d.client.DeletePortMapping("", port, "TCP")
time.Sleep(time.Millisecond)
udpErr := d.client.DeletePortMapping("", port, "UDP")
// only return an error if both deletions failed
if tcpErr != nil && udpErr != nil {
return tcpErr
}
return nil
}
// Location returns the URL of the router, for future lookups (see Load).
func (d *IGD) Location() string {
return d.client.GetServiceClient().Location.String()
}
// getInternalIP returns the user's local IP.
func (d *IGD) getInternalIP() (string, error) {
host, _, _ := net.SplitHostPort(d.client.GetServiceClient().RootDevice.URLBase.Host)
devIP := net.ParseIP(host)
if devIP == nil {
return "", errors.New("could not determine router's internal IP")
}
ifaces, err := net.Interfaces()
if err != nil {
return "", err
}
for _, iface := range ifaces {
addrs, err := iface.Addrs()
if err != nil {
return "", err
}
for _, addr := range addrs {
if x, ok := addr.(*net.IPNet); ok && x.Contains(devIP) {
return x.IP.String(), nil
}
}
}
return "", errors.New("could not determine internal IP")
}
// Discover is deprecated; use DiscoverCtx instead.
func Discover() (*IGD, error) {
return DiscoverCtx(context.Background())
}
// DiscoverCtx scans the local network for routers and returns the first
// UPnP-enabled router it encounters. It will try up to 3 times to find a
// router, sleeping a random duration between each attempt. This is to
// mitigate a race condition with many callers attempting to discover
// simultaneously.
func DiscoverCtx(ctx context.Context) (*IGD, error) {
// TODO: if more than one client is found, only return those on the same
// subnet as the user?
maxTries := 3
sleepTime := time.Millisecond * time.Duration(fastrand.Intn(5000))
for try := 0; try < maxTries; try++ {
pppclients, _, _ := internetgateway1.NewWANPPPConnection1Clients(ctx)
if len(pppclients) > 0 {
return &IGD{pppclients[0]}, nil
}
ipclients, _, _ := internetgateway1.NewWANIPConnection1Clients(ctx)
if len(ipclients) > 0 {
return &IGD{ipclients[0]}, nil
}
select {
case <-ctx.Done():
return nil, context.Canceled
case <-time.After(sleepTime):
}
sleepTime *= 2
}
return nil, errors.New("no UPnP-enabled gateway found")
}
// Load connects to the router service specified by rawurl. This is much
// faster than Discover. Generally, Load should only be called with values
// returned by the IGD's Location method.
func Load(rawurl string) (*IGD, error) {
loc, err := url.Parse(rawurl)
if err != nil {
return nil, err
}
pppclients, _ := internetgateway1.NewWANPPPConnection1ClientsByURL(loc)
if len(pppclients) > 0 {
return &IGD{pppclients[0]}, nil
}
ipclients, _ := internetgateway1.NewWANIPConnection1ClientsByURL(loc)
if len(ipclients) > 0 {
return &IGD{ipclients[0]}, nil
}
return nil, errors.New("no UPnP-enabled gateway found at URL " + rawurl)
}