2007-04-16 21:59:13 +00:00
|
|
|
# portions extracted from ipaddresslib by Autonomous Zone Industries, LGPL (author: Greg Smith)
|
|
|
|
# portions adapted from nattraverso.ipdiscover
|
|
|
|
# portions authored by Brian Warner, working for Allmydata
|
|
|
|
# most recent version authored by Zooko O'Whielacronx, working for Allmydata
|
2006-11-30 23:23:39 +00:00
|
|
|
|
2007-04-16 21:59:13 +00:00
|
|
|
# from the Python Standard Library
|
2007-08-09 17:56:47 +00:00
|
|
|
import os, re, socket, sys
|
2006-11-30 23:23:39 +00:00
|
|
|
|
2007-04-16 21:59:13 +00:00
|
|
|
# from Twisted
|
|
|
|
from twisted.internet import defer
|
2006-11-30 23:23:39 +00:00
|
|
|
from twisted.internet import reactor
|
|
|
|
from twisted.internet.protocol import DatagramProtocol
|
2007-03-08 01:22:30 +00:00
|
|
|
from twisted.internet.utils import getProcessOutput
|
2007-04-16 21:59:13 +00:00
|
|
|
from twisted.python.procutils import which
|
2007-03-08 01:22:30 +00:00
|
|
|
|
2007-08-09 17:56:47 +00:00
|
|
|
# from allmydata.util
|
|
|
|
import observer
|
|
|
|
|
2007-04-16 21:59:13 +00:00
|
|
|
def get_local_addresses_async(target='A.ROOT-SERVERS.NET'):
|
|
|
|
"""
|
|
|
|
Return a Deferred that fires with a list of IPv4 addresses (as dotted-quad
|
2007-05-22 21:06:37 +00:00
|
|
|
strings) that are currently configured on this host, sorted in descending
|
|
|
|
order of how likely we think they are to work.
|
2007-03-08 03:03:47 +00:00
|
|
|
|
2007-04-16 21:59:13 +00:00
|
|
|
@param target: we want to learn an IP address they could try using to
|
|
|
|
connect to us; The default value is fine, but it might help if you
|
|
|
|
pass the address of a host that you are actually trying to be
|
|
|
|
reachable to.
|
2007-03-08 01:22:30 +00:00
|
|
|
"""
|
2007-05-22 21:06:37 +00:00
|
|
|
addresses = []
|
2007-06-08 02:23:33 +00:00
|
|
|
local_ip = get_local_ip_for(target)
|
|
|
|
if local_ip:
|
|
|
|
addresses.append(local_ip)
|
2007-04-19 00:30:08 +00:00
|
|
|
|
2007-05-22 21:06:37 +00:00
|
|
|
if sys.platform == "cygwin":
|
|
|
|
d = _cygwin_hack_find_addresses(target)
|
|
|
|
else:
|
|
|
|
d = _find_addresses_via_config()
|
2007-03-08 01:43:17 +00:00
|
|
|
|
2007-04-16 21:59:13 +00:00
|
|
|
def _collect(res):
|
2007-05-22 21:06:37 +00:00
|
|
|
for addr in res:
|
2007-10-13 07:38:16 +00:00
|
|
|
if addr != "0.0.0.0" and not addr in addresses:
|
2007-05-22 21:06:37 +00:00
|
|
|
addresses.append(addr)
|
2007-03-08 01:22:30 +00:00
|
|
|
return addresses
|
2007-04-16 21:59:13 +00:00
|
|
|
d.addCallback(_collect)
|
2007-03-08 01:22:30 +00:00
|
|
|
|
2007-04-16 21:59:13 +00:00
|
|
|
return d
|
2006-11-30 23:23:39 +00:00
|
|
|
|
2007-04-16 21:59:13 +00:00
|
|
|
def get_local_ip_for(target):
|
2006-11-30 23:23:39 +00:00
|
|
|
"""Find out what our IP address is for use by a given target.
|
|
|
|
|
2007-08-09 17:56:47 +00:00
|
|
|
@return: the IP address as a dotted-quad string which could be used by
|
2007-06-08 02:23:33 +00:00
|
|
|
to connect to us. It might work for them, it might not. If
|
|
|
|
there is no suitable address (perhaps we don't currently have an
|
|
|
|
externally-visible interface), this will return None.
|
2006-11-30 23:23:39 +00:00
|
|
|
"""
|
2007-06-08 02:23:33 +00:00
|
|
|
|
|
|
|
try:
|
|
|
|
target_ipaddr = socket.gethostbyname(target)
|
|
|
|
except socket.gaierror:
|
|
|
|
# DNS isn't running
|
|
|
|
return None
|
2006-11-30 23:23:39 +00:00
|
|
|
udpprot = DatagramProtocol()
|
|
|
|
port = reactor.listenUDP(0, udpprot)
|
2007-06-08 02:23:33 +00:00
|
|
|
try:
|
|
|
|
udpprot.transport.connect(target_ipaddr, 7)
|
|
|
|
localip = udpprot.transport.getHost().host
|
|
|
|
except socket.error:
|
|
|
|
# no route to that host
|
|
|
|
localip = None
|
2006-12-03 02:37:31 +00:00
|
|
|
port.stopListening() # note, this returns a Deferred
|
2006-11-30 23:23:39 +00:00
|
|
|
return localip
|
2006-12-03 02:37:31 +00:00
|
|
|
|
2007-04-16 21:59:13 +00:00
|
|
|
# k: result of sys.platform, v: which kind of IP configuration reader we use
|
|
|
|
_platform_map = {
|
|
|
|
"linux-i386": "linux", # redhat
|
|
|
|
"linux-ppc": "linux", # redhat
|
|
|
|
"linux2": "linux", # debian
|
|
|
|
"win32": "win32",
|
|
|
|
"irix6-n32": "irix",
|
|
|
|
"irix6-n64": "irix",
|
|
|
|
"irix6": "irix",
|
|
|
|
"openbsd2": "bsd",
|
|
|
|
"darwin": "bsd", # Mac OS X
|
|
|
|
"freebsd4": "bsd",
|
|
|
|
"freebsd5": "bsd",
|
|
|
|
"netbsd1": "bsd",
|
|
|
|
"sunos5": "sunos",
|
|
|
|
"cygwin": "cygwin",
|
|
|
|
}
|
|
|
|
|
|
|
|
class UnsupportedPlatformError(Exception):
|
|
|
|
pass
|
|
|
|
|
|
|
|
# Wow, I'm really amazed at home much mileage we've gotten out of calling
|
|
|
|
# the external route.exe program on windows... It appears to work on all
|
|
|
|
# versions so far. Still, the real system calls would much be preferred...
|
|
|
|
# ... thus wrote Greg Smith in time immemorial...
|
|
|
|
_win32_path = 'route.exe'
|
|
|
|
_win32_args = ('print',)
|
2007-04-16 22:12:01 +00:00
|
|
|
_win32_re = re.compile('^\s*\d+\.\d+\.\d+\.\d+\s.+\s(?P<address>\d+\.\d+\.\d+\.\d+)\s+(?P<metric>\d+)\s*$', flags=re.M|re.I|re.S)
|
2007-04-16 21:59:13 +00:00
|
|
|
|
|
|
|
# These work in Redhat 6.x and Debian 2.2 potato
|
|
|
|
_linux_path = '/sbin/ifconfig'
|
2007-04-16 22:12:01 +00:00
|
|
|
_linux_re = re.compile('^\s*inet addr:(?P<address>\d+\.\d+\.\d+\.\d+)\s.+$', flags=re.M|re.I|re.S)
|
2007-04-16 21:59:13 +00:00
|
|
|
|
|
|
|
# NetBSD 1.4 (submitted by Rhialto), Darwin, Mac OS X
|
2007-04-18 14:40:26 +00:00
|
|
|
_netbsd_path = '/sbin/ifconfig'
|
|
|
|
_netbsd_args = ('-a',)
|
2007-04-16 21:59:13 +00:00
|
|
|
_netbsd_re = re.compile('^\s+inet (?P<address>\d+\.\d+\.\d+\.\d+)\s.+$', flags=re.M|re.I|re.S)
|
|
|
|
|
|
|
|
# Irix 6.5
|
2007-04-18 14:40:26 +00:00
|
|
|
_irix_path = '/usr/etc/ifconfig'
|
2007-04-16 21:59:13 +00:00
|
|
|
|
|
|
|
# Solaris 2.x
|
2007-04-18 14:40:26 +00:00
|
|
|
_sunos_path = '/usr/sbin/ifconfig'
|
2007-04-16 21:59:13 +00:00
|
|
|
|
2007-08-09 17:56:47 +00:00
|
|
|
class SequentialTrier(object):
|
|
|
|
""" I hold a list of executables to try and try each one in turn
|
|
|
|
until one gives me a list of IP addresses."""
|
|
|
|
|
|
|
|
def __init__(self, exebasename, args, regex):
|
|
|
|
assert not os.path.isabs(exebasename)
|
|
|
|
self.exes_left_to_try = which(exebasename)
|
|
|
|
self.exes_left_to_try.reverse()
|
|
|
|
self.args = args
|
|
|
|
self.regex = regex
|
|
|
|
self.o = observer.OneShotObserverList()
|
|
|
|
self._try_next()
|
|
|
|
|
|
|
|
def _try_next(self):
|
|
|
|
if not self.exes_left_to_try:
|
|
|
|
self.o.fire(None)
|
|
|
|
else:
|
|
|
|
exe = self.exes_left_to_try.pop()
|
|
|
|
d2 = _query(exe, self.args, self.regex)
|
|
|
|
|
|
|
|
def cb(res):
|
|
|
|
if res:
|
|
|
|
self.o.fire(res)
|
|
|
|
else:
|
|
|
|
self._try_next()
|
|
|
|
|
|
|
|
def eb(why):
|
|
|
|
self._try_next()
|
|
|
|
|
|
|
|
d2.addCallbacks(cb, eb)
|
|
|
|
|
|
|
|
def when_tried(self):
|
|
|
|
return self.o.when_fired()
|
|
|
|
|
2007-04-19 00:30:08 +00:00
|
|
|
# k: platform string as provided in the value of _platform_map
|
|
|
|
# v: tuple of (path_to_tool, args, regex,)
|
|
|
|
_tool_map = {
|
|
|
|
"linux": (_linux_path, (), _linux_re,),
|
|
|
|
"win32": (_win32_path, _win32_args, _win32_re,),
|
|
|
|
"cygwin": (_win32_path, _win32_args, _win32_re,),
|
|
|
|
"bsd": (_netbsd_path, _netbsd_args, _netbsd_re,),
|
|
|
|
"irix": (_irix_path, _netbsd_args, _netbsd_re,),
|
|
|
|
"sunos": (_sunos_path, _netbsd_args, _netbsd_re,),
|
|
|
|
}
|
2007-04-16 21:59:13 +00:00
|
|
|
def _find_addresses_via_config():
|
|
|
|
# originally by Greg Smith, hacked by Zooko to conform to Brian's API
|
|
|
|
|
|
|
|
platform = _platform_map.get(sys.platform)
|
|
|
|
if not platform:
|
|
|
|
raise UnsupportedPlatformError(sys.platform)
|
|
|
|
|
2007-04-19 00:30:08 +00:00
|
|
|
(pathtotool, args, regex,) = _tool_map[platform]
|
|
|
|
|
2007-08-09 17:56:47 +00:00
|
|
|
# If pathtotool is a fully qualified path then we just try that.
|
|
|
|
# If it is merely an executable name then we use Twisted's
|
|
|
|
# "which()" utility and try each executable in turn until one
|
|
|
|
# gives us something that resembles a dotted-quad IPv4 address.
|
|
|
|
|
|
|
|
if os.path.isabs(pathtotool):
|
|
|
|
return _query(pathtotool, args, regex)
|
|
|
|
else:
|
|
|
|
return SequentialTrier(pathtotool, args, regex).when_tried()
|
|
|
|
|
2007-04-19 00:30:08 +00:00
|
|
|
def _query(path, args, regex):
|
2007-04-16 21:59:13 +00:00
|
|
|
d = getProcessOutput(path, args)
|
|
|
|
def _parse(output):
|
2007-05-22 21:06:37 +00:00
|
|
|
addresses = []
|
2007-04-16 21:59:13 +00:00
|
|
|
outputsplit = output.split('\n')
|
2007-04-16 22:12:01 +00:00
|
|
|
for outline in outputsplit:
|
|
|
|
m = regex.match(outline)
|
2007-04-16 21:59:13 +00:00
|
|
|
if m:
|
2007-05-22 21:06:37 +00:00
|
|
|
addr = m.groupdict()['address']
|
|
|
|
if addr not in addresses:
|
|
|
|
addresses.append(addr)
|
2007-04-16 21:59:13 +00:00
|
|
|
|
|
|
|
return addresses
|
|
|
|
d.addCallback(_parse)
|
|
|
|
return d
|
2007-04-19 00:30:08 +00:00
|
|
|
|
2007-05-22 21:06:37 +00:00
|
|
|
def _cygwin_hack_find_addresses(target):
|
|
|
|
addresses = []
|
2007-04-19 00:30:08 +00:00
|
|
|
for h in [target, "localhost", "127.0.0.1",]:
|
|
|
|
try:
|
2007-05-22 21:06:37 +00:00
|
|
|
addr = get_local_ip_for(h)
|
|
|
|
if addr not in addresses:
|
|
|
|
addresses.append(addr)
|
2007-04-19 00:30:08 +00:00
|
|
|
except socket.gaierror:
|
|
|
|
pass
|
|
|
|
|
2007-05-22 21:06:37 +00:00
|
|
|
return defer.succeed(addresses)
|