mirror of
https://github.com/status-im/status-go.git
synced 2025-01-10 14:47:06 +00:00
eeca435064
Update vendor Integrate rendezvous into status node Add a test with failover using rendezvous Use multiple servers in client Use discovery V5 by default and test that node can be started with rendezvous discovet Fix linter Update rendezvous client to one with instrumented stream Address feedback Fix test with updated topic limits Apply several suggestions Change log to debug for request errors because we continue execution Remove web3js after rebase Update rendezvous package
240 lines
5.5 KiB
Go
240 lines
5.5 KiB
Go
package nat
|
|
|
|
import (
|
|
"net"
|
|
"time"
|
|
|
|
"github.com/huin/goupnp"
|
|
"github.com/huin/goupnp/dcps/internetgateway1"
|
|
"github.com/huin/goupnp/dcps/internetgateway2"
|
|
)
|
|
|
|
var (
|
|
_ NAT = (*upnp_NAT)(nil)
|
|
)
|
|
|
|
func discoverUPNP_IG1() <-chan NAT {
|
|
res := make(chan NAT, 1)
|
|
go func() {
|
|
|
|
// find devices
|
|
devs, err := goupnp.DiscoverDevices(internetgateway1.URN_WANConnectionDevice_1)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
for _, dev := range devs {
|
|
if dev.Root == nil {
|
|
continue
|
|
}
|
|
|
|
dev.Root.Device.VisitServices(func(srv *goupnp.Service) {
|
|
switch srv.ServiceType {
|
|
case internetgateway1.URN_WANIPConnection_1:
|
|
client := &internetgateway1.WANIPConnection1{ServiceClient: goupnp.ServiceClient{
|
|
SOAPClient: srv.NewSOAPClient(),
|
|
RootDevice: dev.Root,
|
|
Service: srv,
|
|
}}
|
|
_, isNat, err := client.GetNATRSIPStatus()
|
|
if err == nil && isNat {
|
|
res <- &upnp_NAT{client, make(map[int]int), "UPNP (IG1-IP1)", dev.Root}
|
|
return
|
|
}
|
|
|
|
case internetgateway1.URN_WANPPPConnection_1:
|
|
client := &internetgateway1.WANPPPConnection1{ServiceClient: goupnp.ServiceClient{
|
|
SOAPClient: srv.NewSOAPClient(),
|
|
RootDevice: dev.Root,
|
|
Service: srv,
|
|
}}
|
|
_, isNat, err := client.GetNATRSIPStatus()
|
|
if err == nil && isNat {
|
|
res <- &upnp_NAT{client, make(map[int]int), "UPNP (IG1-PPP1)", dev.Root}
|
|
return
|
|
}
|
|
|
|
}
|
|
})
|
|
}
|
|
|
|
}()
|
|
return res
|
|
}
|
|
|
|
func discoverUPNP_IG2() <-chan NAT {
|
|
res := make(chan NAT, 1)
|
|
go func() {
|
|
|
|
// find devices
|
|
devs, err := goupnp.DiscoverDevices(internetgateway2.URN_WANConnectionDevice_2)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
for _, dev := range devs {
|
|
if dev.Root == nil {
|
|
continue
|
|
}
|
|
|
|
dev.Root.Device.VisitServices(func(srv *goupnp.Service) {
|
|
switch srv.ServiceType {
|
|
case internetgateway2.URN_WANIPConnection_1:
|
|
client := &internetgateway2.WANIPConnection1{ServiceClient: goupnp.ServiceClient{
|
|
SOAPClient: srv.NewSOAPClient(),
|
|
RootDevice: dev.Root,
|
|
Service: srv,
|
|
}}
|
|
_, isNat, err := client.GetNATRSIPStatus()
|
|
if err == nil && isNat {
|
|
res <- &upnp_NAT{client, make(map[int]int), "UPNP (IG2-IP1)", dev.Root}
|
|
return
|
|
}
|
|
|
|
case internetgateway2.URN_WANIPConnection_2:
|
|
client := &internetgateway2.WANIPConnection2{ServiceClient: goupnp.ServiceClient{
|
|
SOAPClient: srv.NewSOAPClient(),
|
|
RootDevice: dev.Root,
|
|
Service: srv,
|
|
}}
|
|
_, isNat, err := client.GetNATRSIPStatus()
|
|
if err == nil && isNat {
|
|
res <- &upnp_NAT{client, make(map[int]int), "UPNP (IG2-IP2)", dev.Root}
|
|
return
|
|
}
|
|
|
|
case internetgateway2.URN_WANPPPConnection_1:
|
|
client := &internetgateway2.WANPPPConnection1{ServiceClient: goupnp.ServiceClient{
|
|
SOAPClient: srv.NewSOAPClient(),
|
|
RootDevice: dev.Root,
|
|
Service: srv,
|
|
}}
|
|
_, isNat, err := client.GetNATRSIPStatus()
|
|
if err == nil && isNat {
|
|
res <- &upnp_NAT{client, make(map[int]int), "UPNP (IG2-PPP1)", dev.Root}
|
|
return
|
|
}
|
|
|
|
}
|
|
})
|
|
}
|
|
|
|
}()
|
|
return res
|
|
}
|
|
|
|
type upnp_NAT_Client interface {
|
|
GetExternalIPAddress() (string, error)
|
|
AddPortMapping(string, uint16, string, uint16, string, bool, string, uint32) error
|
|
DeletePortMapping(string, uint16, string) error
|
|
}
|
|
|
|
type upnp_NAT struct {
|
|
c upnp_NAT_Client
|
|
ports map[int]int
|
|
typ string
|
|
rootDevice *goupnp.RootDevice
|
|
}
|
|
|
|
func (u *upnp_NAT) GetExternalAddress() (addr net.IP, err error) {
|
|
ipString, err := u.c.GetExternalIPAddress()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
ip := net.ParseIP(ipString)
|
|
if ip == nil {
|
|
return nil, ErrNoExternalAddress
|
|
}
|
|
|
|
return ip, nil
|
|
}
|
|
|
|
func mapProtocol(s string) string {
|
|
switch s {
|
|
case "udp":
|
|
return "UDP"
|
|
case "tcp":
|
|
return "TCP"
|
|
default:
|
|
panic("invalid protocol: " + s)
|
|
}
|
|
}
|
|
|
|
func (u *upnp_NAT) AddPortMapping(protocol string, internalPort int, description string, timeout time.Duration) (int, error) {
|
|
ip, err := u.GetInternalAddress()
|
|
if err != nil {
|
|
return 0, nil
|
|
}
|
|
|
|
timeoutInSeconds := uint32(timeout / time.Second)
|
|
|
|
if externalPort := u.ports[internalPort]; externalPort > 0 {
|
|
err = u.c.AddPortMapping("", uint16(externalPort), mapProtocol(protocol), uint16(internalPort), ip.String(), true, description, timeoutInSeconds)
|
|
if err == nil {
|
|
return externalPort, nil
|
|
}
|
|
}
|
|
|
|
for i := 0; i < 3; i++ {
|
|
externalPort := randomPort()
|
|
err = u.c.AddPortMapping("", uint16(externalPort), mapProtocol(protocol), uint16(internalPort), ip.String(), true, description, timeoutInSeconds)
|
|
if err == nil {
|
|
u.ports[internalPort] = externalPort
|
|
return externalPort, nil
|
|
}
|
|
}
|
|
|
|
return 0, err
|
|
}
|
|
|
|
func (u *upnp_NAT) DeletePortMapping(protocol string, internalPort int) error {
|
|
if externalPort := u.ports[internalPort]; externalPort > 0 {
|
|
delete(u.ports, internalPort)
|
|
return u.c.DeletePortMapping("", uint16(externalPort), mapProtocol(protocol))
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (u *upnp_NAT) GetDeviceAddress() (net.IP, error) {
|
|
addr, err := net.ResolveUDPAddr("udp4", u.rootDevice.URLBase.Host)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return addr.IP, nil
|
|
}
|
|
|
|
func (u *upnp_NAT) GetInternalAddress() (net.IP, error) {
|
|
devAddr, err := u.GetDeviceAddress()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
ifaces, err := net.Interfaces()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
for _, iface := range ifaces {
|
|
addrs, err := iface.Addrs()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
for _, addr := range addrs {
|
|
switch x := addr.(type) {
|
|
case *net.IPNet:
|
|
if x.Contains(devAddr) {
|
|
return x.IP, nil
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return nil, ErrNoInternalAddress
|
|
}
|
|
|
|
func (n *upnp_NAT) Type() string { return n.typ }
|