Commit 3a9e668c authored by Julien Muchembled's avatar Julien Muchembled

UPnP: randomize external port

Some routers are so broken that UPnP NAT don't report ConflictInMappingEntry
when redirecting the same port several times.

Here is for example what we had with a Numericable Box (France):

0 (1024, 'TCP', ('192.168.0.29', 1194), 're6stnet openvpn server (1194/tcp)', '1', '', 0)
1 (1024, 'TCP', ('192.168.0.16', 1194), 're6stnet openvpn server (1194/tcp)', '1', '', 0)
2 (1024, 'TCP', ('192.168.0.33', 1194), 're6stnet openvpn server (1194/tcp)', '1', '', 0)
3 (1024, 'TCP', ('192.168.0.20', 1194), 're6stnet openvpn server (1194/tcp)', '1', '', 0)
('192.168.0.29', 1194, 're6stnet openvpn server (1194/tcp)', True, 0)

Obviously, this can't work.

It seems that this router also accepts a limited number of NAT rules, far less
than we'd like, so even if there's still a probability of conflict with this
commit, it will be good enough for our use.
parent e3c42494
from functools import wraps
import logging, socket, time
import logging, random, socket, time
import miniupnpc
......@@ -8,10 +8,18 @@ class UPnPException(Exception):
class Forwarder(object):
"""
External port is chosen randomly between 32768 & 49151 included.
"""
next_refresh = 0
_next_retry = -1
_next_port = 1024
_next_port = random.randrange(0, 8192)
@classmethod
def _getExternalPort(cls):
port = cls._next_port = (cls._next_port + 1) % 8192
return 32768 + port
def __init__(self, description):
self._description = description
......@@ -73,6 +81,7 @@ class Forwarder(object):
logging.debug('Refreshing port forwarding')
ip = self.externalipaddress()
lanaddr = self._u.lanaddr
retry = 8191
for r in self._rules:
local, proto, port = r
if port and not force:
......@@ -81,11 +90,11 @@ class Forwarder(object):
args = proto.upper(), lanaddr, local, desc, ''
while True:
if port is None:
port = self._next_port
if port > 65535:
if not retry:
raise UPnPException('No free port to redirect %s'
% desc)
self._next_port = port + 1
retry -= 1
port = self._getExternalPort()
try:
self.addportmapping(port, *args)
break
......@@ -99,10 +108,6 @@ class Forwarder(object):
return ip
def clear(self):
try:
del self._next_port
except AttributeError:
return
for r in self._rules:
port = r[2]
if port:
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment