2016-03-03 20:24:07 +00:00
|
|
|
from __future__ import print_function
|
|
|
|
import time
|
|
|
|
from zope.interface import implementer
|
|
|
|
from twisted.web import error as web_error
|
|
|
|
from twisted.internet.defer import inlineCallbacks, returnValue
|
|
|
|
from twisted.python.compat import nativeString
|
|
|
|
from twisted.internet.error import ConnectError
|
|
|
|
from twisted.web import iweb
|
|
|
|
import txtorcon
|
|
|
|
import ipaddr
|
2016-04-16 00:48:44 +00:00
|
|
|
from wormhole.timing import DebugTiming
|
2016-03-03 20:24:07 +00:00
|
|
|
from .transit import allocate_tcp_port
|
|
|
|
|
|
|
|
# based on twisted.web.client._StandardEndpointFactory
|
|
|
|
@implementer(iweb.IAgentEndpointFactory)
|
|
|
|
class TorWebAgentEndpointFactory(object):
|
|
|
|
def __init__(self, reactor, socks_port):
|
|
|
|
self._reactor = reactor
|
|
|
|
self._socks_port = socks_port
|
|
|
|
|
|
|
|
def endpointForURI(self, uri):
|
|
|
|
try:
|
|
|
|
host = nativeString(uri.host)
|
|
|
|
except UnicodeDecodeError:
|
|
|
|
raise ValueError(("The host of the provided URI ({uri.host!r}) "
|
|
|
|
"contains non-ASCII octets, it should be ASCII "
|
|
|
|
"decodable.").format(uri=uri))
|
|
|
|
|
2016-04-20 19:05:27 +00:00
|
|
|
if uri.scheme == b'http' or uri.scheme == b'ws':
|
2016-03-03 20:24:07 +00:00
|
|
|
print("building URI endpoint with tor for %s" % uri.toBytes())
|
|
|
|
return txtorcon.TorClientEndpoint(#self._reactor,
|
|
|
|
host, uri.port,
|
|
|
|
socks_hostname="127.0.0.1", socks_port=self._socks_port)
|
|
|
|
elif uri.scheme == b'https':
|
|
|
|
raise NotImplementedError
|
|
|
|
# find some twisted thing that wraps a normal
|
|
|
|
# IStreamClientEndpoint in a TLS-ifying layer, and wrap it around
|
|
|
|
# a TorClientEndpoint. Maybe t.i.endpoints.wrapClientTLS
|
|
|
|
else:
|
|
|
|
raise web_error.SchemeNotSupported("Unsupported scheme: %r" % (uri.scheme,))
|
|
|
|
|
|
|
|
class TorManager:
|
2016-03-03 22:31:13 +00:00
|
|
|
def __init__(self, reactor, tor_socks_port=None, tor_control_port=9051,
|
|
|
|
timing=None):
|
2016-03-03 20:24:07 +00:00
|
|
|
"""
|
|
|
|
If tor_socks_port= is provided, I will assume that it points to a
|
|
|
|
functioning SOCKS server, and will use it for all outbound
|
|
|
|
connections. I will not attempt to establish a control-port
|
|
|
|
connection, and I will not be able to run a server.
|
|
|
|
|
|
|
|
Otherwise, I will try to connect to an existing Tor process, first on
|
|
|
|
localhost:9051, then /var/run/tor/control. Then I will try to
|
|
|
|
authenticate, by reading a cookie file named by the Tor process. This
|
|
|
|
will succeed if 1: Tor is already running, and 2: the current user
|
|
|
|
can read that file (either they started it, e.g. TorBrowser, or they
|
|
|
|
are in a unix group that's been given access, e.g. debian-tor).
|
|
|
|
|
|
|
|
If tor_control_port= is provided, I will use it instead of 9051.
|
|
|
|
"""
|
|
|
|
self._reactor = reactor
|
|
|
|
# note: False is int
|
|
|
|
assert isinstance(tor_socks_port, (int, type(None)))
|
|
|
|
assert isinstance(tor_control_port, int)
|
|
|
|
self._tor_socks_port = tor_socks_port
|
|
|
|
self._tor_control_port = tor_control_port
|
2016-03-03 22:31:13 +00:00
|
|
|
self._timing = timing or DebugTiming()
|
2016-03-03 20:24:07 +00:00
|
|
|
|
|
|
|
@inlineCallbacks
|
|
|
|
def start(self):
|
|
|
|
# Connect to an existing Tor, or create a new one. If we need to
|
|
|
|
# launch an onion service, then we need a working control port (and
|
|
|
|
# authentication cookie). If we're only acting as a client, we don't
|
|
|
|
# need the control port.
|
|
|
|
|
|
|
|
if self._tor_socks_port is not None:
|
|
|
|
self._can_run_service = False
|
|
|
|
returnValue(True)
|
|
|
|
|
2016-03-03 22:31:13 +00:00
|
|
|
_start_find = self._timing.add_event("find tor")
|
2016-03-03 20:24:07 +00:00
|
|
|
# try port 9051, then try /var/run/tor/control . Throws on failure.
|
|
|
|
state = None
|
2016-03-03 23:07:05 +00:00
|
|
|
_start_tcp = self._timing.add_event("tor localhost")
|
2016-03-03 20:24:07 +00:00
|
|
|
try:
|
|
|
|
connection = (self._reactor, "127.0.0.1", self._tor_control_port)
|
|
|
|
state = yield txtorcon.build_tor_connection(connection)
|
|
|
|
self._tor_protocol = state.protocol
|
|
|
|
except ConnectError:
|
|
|
|
print("unable to reach Tor on %d" % self._tor_control_port)
|
|
|
|
pass
|
2016-03-03 23:07:05 +00:00
|
|
|
self._timing.finish_event(_start_tcp)
|
2016-03-03 20:24:07 +00:00
|
|
|
|
|
|
|
if not state:
|
2016-03-03 23:07:05 +00:00
|
|
|
_start_unix = self._timing.add_event("tor unix")
|
2016-03-03 20:24:07 +00:00
|
|
|
try:
|
|
|
|
connection = (self._reactor, "/var/run/tor/control")
|
2016-03-24 15:29:23 +00:00
|
|
|
# add build_state=False to get back a Protocol object instead
|
|
|
|
# of a State object
|
2016-03-03 20:24:07 +00:00
|
|
|
state = yield txtorcon.build_tor_connection(connection)
|
|
|
|
self._tor_protocol = state.protocol
|
|
|
|
except (ValueError, ConnectError):
|
|
|
|
print("unable to reach Tor on /var/run/tor/control")
|
|
|
|
pass
|
2016-03-03 23:07:05 +00:00
|
|
|
self._timing.finish_event(_start_unix)
|
2016-03-03 20:24:07 +00:00
|
|
|
|
|
|
|
if state:
|
|
|
|
print("connected to pre-existing Tor process")
|
|
|
|
print("state:", state)
|
|
|
|
else:
|
|
|
|
print("launching my own Tor process")
|
|
|
|
yield self._create_my_own_tor()
|
|
|
|
# that sets self._tor_socks_port and self._tor_protocol
|
|
|
|
|
2016-03-03 22:31:13 +00:00
|
|
|
self._timing.finish_event(_start_find)
|
2016-03-03 20:24:07 +00:00
|
|
|
self._can_run_service = True
|
|
|
|
returnValue(True)
|
|
|
|
|
|
|
|
@inlineCallbacks
|
|
|
|
def _create_my_own_tor(self):
|
2016-03-03 22:31:13 +00:00
|
|
|
_start_launch = self._timing.add_event("launch tor")
|
2016-03-03 20:24:07 +00:00
|
|
|
start = time.time()
|
|
|
|
config = self.config = txtorcon.TorConfig()
|
|
|
|
if 0:
|
|
|
|
# The default is for launch_tor to create a tempdir itself, and
|
|
|
|
# delete it when done. We only need to set a DataDirectory if we
|
|
|
|
# want it to be persistent.
|
|
|
|
import tempfile
|
|
|
|
datadir = tempfile.mkdtemp()
|
|
|
|
config.DataDirectory = datadir
|
|
|
|
|
2016-03-24 15:29:23 +00:00
|
|
|
#config.ControlPort = allocate_tcp_port() # defaults to 9052
|
|
|
|
#print("setting config.ControlPort to", config.ControlPort)
|
2016-03-03 20:24:07 +00:00
|
|
|
config.SocksPort = allocate_tcp_port()
|
|
|
|
self._tor_socks_port = config.SocksPort
|
|
|
|
print("setting config.SocksPort to", config.SocksPort)
|
|
|
|
|
|
|
|
tpp = yield txtorcon.launch_tor(config, self._reactor,
|
|
|
|
#tor_binary=
|
|
|
|
)
|
|
|
|
# gives a TorProcessProtocol with .tor_protocol
|
|
|
|
self._tor_protocol = tpp.tor_protocol
|
|
|
|
print("tp:", self._tor_protocol)
|
|
|
|
print("elapsed:", time.time() - start)
|
2016-03-03 22:31:13 +00:00
|
|
|
self._timing.finish_event(_start_launch)
|
2016-03-03 20:24:07 +00:00
|
|
|
returnValue(True)
|
|
|
|
|
|
|
|
def get_web_agent_endpoint_factory(self):
|
|
|
|
return TorWebAgentEndpointFactory(self._reactor, self._tor_socks_port)
|
|
|
|
|
|
|
|
def is_non_public_numeric_address(self, host):
|
|
|
|
# for numeric hostnames, skip RFC1918 addresses, since no Tor exit
|
|
|
|
# node will be able to reach those. Likewise ignore IPv6 addresses.
|
|
|
|
try:
|
|
|
|
a = ipaddr.IPAddress(host)
|
|
|
|
except ValueError:
|
|
|
|
return False # non-numeric, let Tor try it
|
|
|
|
if a.version != 4:
|
2016-03-24 15:29:23 +00:00
|
|
|
return True # IPv6 gets ignored
|
2016-03-03 20:24:07 +00:00
|
|
|
if (a.is_loopback or a.is_multicast or a.is_private or a.is_reserved
|
|
|
|
or a.is_unspecified):
|
|
|
|
return True # too weird, don't connect
|
|
|
|
return False
|
|
|
|
|
|
|
|
def get_endpoint_for(self, host, port):
|
|
|
|
assert isinstance(port, int)
|
|
|
|
if self.is_non_public_numeric_address(host):
|
|
|
|
print("ignoring non-Tor-able %s" % host)
|
|
|
|
return None
|
|
|
|
|
|
|
|
# txsocksx doesn't like unicode: it concatenates some binary protocol
|
|
|
|
# bytes with the hostname when talking to the SOCKS server, so the
|
|
|
|
# py2 automatic unicode promotion blows up
|
|
|
|
host = host.encode("ascii")
|
|
|
|
ep = txtorcon.TorClientEndpoint(host, port,
|
|
|
|
socks_hostname="127.0.0.1",
|
|
|
|
socks_port=self._tor_socks_port)
|
|
|
|
return ep
|