2016-10-09 04:57:25 +00:00
|
|
|
import os
|
|
|
|
from twisted.trial import unittest
|
|
|
|
from twisted.internet import defer, error
|
|
|
|
from StringIO import StringIO
|
|
|
|
import mock
|
|
|
|
from ..util import tor_provider
|
|
|
|
from ..scripts import create_node, runner
|
|
|
|
from foolscap.eventual import flushEventualQueue
|
|
|
|
|
|
|
|
def mock_txtorcon(txtorcon):
|
|
|
|
return mock.patch("allmydata.util.tor_provider._import_txtorcon",
|
|
|
|
return_value=txtorcon)
|
|
|
|
|
|
|
|
def mock_tor(tor):
|
|
|
|
return mock.patch("allmydata.util.tor_provider._import_tor",
|
|
|
|
return_value=tor)
|
|
|
|
|
|
|
|
def make_cli_config(basedir, *argv):
|
|
|
|
parent = runner.Options()
|
|
|
|
cli_config = create_node.CreateNodeOptions()
|
|
|
|
cli_config.parent = parent
|
|
|
|
cli_config.parseOptions(argv)
|
|
|
|
cli_config["basedir"] = basedir
|
|
|
|
cli_config.stdout = StringIO()
|
|
|
|
return cli_config
|
|
|
|
|
|
|
|
class TryToConnect(unittest.TestCase):
|
|
|
|
def test_try(self):
|
|
|
|
reactor = object()
|
|
|
|
txtorcon = mock.Mock()
|
|
|
|
tor_state = object()
|
|
|
|
d = defer.succeed(tor_state)
|
|
|
|
txtorcon.build_tor_connection = mock.Mock(return_value=d)
|
|
|
|
ep = object()
|
|
|
|
stdout = StringIO()
|
|
|
|
with mock.patch("allmydata.util.tor_provider.clientFromString",
|
|
|
|
return_value=ep) as cfs:
|
|
|
|
d = tor_provider._try_to_connect(reactor, "desc", stdout, txtorcon)
|
|
|
|
r = self.successResultOf(d)
|
|
|
|
self.assertIs(r, tor_state)
|
|
|
|
cfs.assert_called_with(reactor, "desc")
|
|
|
|
txtorcon.build_tor_connection.assert_called_with(ep)
|
|
|
|
|
|
|
|
def test_try_handled_error(self):
|
|
|
|
reactor = object()
|
|
|
|
txtorcon = mock.Mock()
|
|
|
|
d = defer.fail(error.ConnectError("oops"))
|
|
|
|
txtorcon.build_tor_connection = mock.Mock(return_value=d)
|
|
|
|
ep = object()
|
|
|
|
stdout = StringIO()
|
|
|
|
with mock.patch("allmydata.util.tor_provider.clientFromString",
|
|
|
|
return_value=ep) as cfs:
|
|
|
|
d = tor_provider._try_to_connect(reactor, "desc", stdout, txtorcon)
|
|
|
|
r = self.successResultOf(d)
|
|
|
|
self.assertIs(r, None)
|
|
|
|
cfs.assert_called_with(reactor, "desc")
|
|
|
|
txtorcon.build_tor_connection.assert_called_with(ep)
|
|
|
|
self.assertEqual(stdout.getvalue(),
|
|
|
|
"Unable to reach Tor at 'desc': "
|
|
|
|
"An error occurred while connecting: oops.\n")
|
|
|
|
|
|
|
|
def test_try_unhandled_error(self):
|
|
|
|
reactor = object()
|
|
|
|
txtorcon = mock.Mock()
|
|
|
|
d = defer.fail(ValueError("oops"))
|
|
|
|
txtorcon.build_tor_connection = mock.Mock(return_value=d)
|
|
|
|
ep = object()
|
|
|
|
stdout = StringIO()
|
|
|
|
with mock.patch("allmydata.util.tor_provider.clientFromString",
|
|
|
|
return_value=ep) as cfs:
|
|
|
|
d = tor_provider._try_to_connect(reactor, "desc", stdout, txtorcon)
|
|
|
|
f = self.failureResultOf(d)
|
|
|
|
self.assertIsInstance(f.value, ValueError)
|
|
|
|
self.assertEqual(str(f.value), "oops")
|
|
|
|
cfs.assert_called_with(reactor, "desc")
|
|
|
|
txtorcon.build_tor_connection.assert_called_with(ep)
|
|
|
|
self.assertEqual(stdout.getvalue(), "")
|
|
|
|
|
|
|
|
class LaunchTor(unittest.TestCase):
|
|
|
|
def _do_test_launch(self, tor_executable):
|
|
|
|
reactor = object()
|
|
|
|
private_dir = "private"
|
|
|
|
txtorcon = mock.Mock()
|
|
|
|
tpp = mock.Mock
|
|
|
|
tpp.tor_protocol = mock.Mock()
|
|
|
|
txtorcon.launch_tor = mock.Mock(return_value=tpp)
|
|
|
|
|
|
|
|
with mock.patch("allmydata.util.tor_provider.allocate_tcp_port",
|
|
|
|
return_value=999999):
|
|
|
|
d = tor_provider._launch_tor(reactor, tor_executable, private_dir,
|
|
|
|
txtorcon)
|
|
|
|
tor_control_endpoint, tor_control_proto = self.successResultOf(d)
|
|
|
|
self.assertIs(tor_control_proto, tpp.tor_protocol)
|
|
|
|
|
|
|
|
def test_launch(self):
|
|
|
|
return self._do_test_launch(None)
|
|
|
|
def test_launch_executable(self):
|
|
|
|
return self._do_test_launch("mytor")
|
|
|
|
|
|
|
|
class ConnectToTor(unittest.TestCase):
|
|
|
|
def _do_test_connect(self, endpoint, reachable):
|
|
|
|
reactor = object()
|
|
|
|
txtorcon = object()
|
|
|
|
args = []
|
|
|
|
if endpoint:
|
|
|
|
args = ["--tor-control-port=%s" % endpoint]
|
|
|
|
cli_config = make_cli_config("basedir", "--listen=tor", *args)
|
|
|
|
stdout = cli_config.stdout
|
|
|
|
expected_port = "tcp:127.0.0.1:9151"
|
|
|
|
if endpoint:
|
|
|
|
expected_port = endpoint
|
|
|
|
tor_state = mock.Mock
|
|
|
|
tor_state.protocol = object()
|
|
|
|
tried = []
|
|
|
|
def _try_to_connect(reactor, port, stdout, txtorcon):
|
|
|
|
tried.append( (reactor, port, stdout, txtorcon) )
|
|
|
|
if not reachable:
|
|
|
|
return defer.succeed(None)
|
|
|
|
if port == expected_port: # second one on the list
|
|
|
|
return defer.succeed(tor_state)
|
|
|
|
return defer.succeed(None)
|
|
|
|
|
|
|
|
with mock.patch("allmydata.util.tor_provider._try_to_connect",
|
|
|
|
_try_to_connect):
|
|
|
|
d = tor_provider._connect_to_tor(reactor, cli_config, txtorcon)
|
|
|
|
if not reachable:
|
|
|
|
f = self.failureResultOf(d)
|
|
|
|
self.assertIsInstance(f.value, ValueError)
|
|
|
|
self.assertEqual(str(f.value),
|
|
|
|
"unable to reach any default Tor control port")
|
|
|
|
return
|
|
|
|
successful_port, tor_control_proto = self.successResultOf(d)
|
|
|
|
self.assertEqual(successful_port, expected_port)
|
|
|
|
self.assertIs(tor_control_proto, tor_state.protocol)
|
|
|
|
expected = [(reactor, "unix:/var/run/tor/control", stdout, txtorcon),
|
|
|
|
(reactor, "tcp:127.0.0.1:9051", stdout, txtorcon),
|
|
|
|
(reactor, "tcp:127.0.0.1:9151", stdout, txtorcon),
|
|
|
|
]
|
|
|
|
if endpoint:
|
|
|
|
expected = [(reactor, endpoint, stdout, txtorcon)]
|
|
|
|
self.assertEqual(tried, expected)
|
|
|
|
|
|
|
|
def test_connect(self):
|
|
|
|
return self._do_test_connect(None, True)
|
|
|
|
def test_connect_endpoint(self):
|
|
|
|
return self._do_test_connect("tcp:other:port", True)
|
|
|
|
def test_connect_unreachable(self):
|
|
|
|
return self._do_test_connect(None, False)
|
|
|
|
|
|
|
|
|
|
|
|
class CreateOnion(unittest.TestCase):
|
|
|
|
def test_no_txtorcon(self):
|
|
|
|
with mock.patch("allmydata.util.tor_provider._import_txtorcon",
|
|
|
|
return_value=None):
|
2017-07-19 17:09:48 +00:00
|
|
|
d = tor_provider.create_config("reactor", "cli_config")
|
2016-10-09 04:57:25 +00:00
|
|
|
f = self.failureResultOf(d)
|
|
|
|
self.assertIsInstance(f.value, ValueError)
|
|
|
|
self.assertEqual(str(f.value),
|
|
|
|
"Cannot create onion without txtorcon. "
|
|
|
|
"Please 'pip install tahoe-lafs[tor]' to fix this.")
|
|
|
|
def _do_test_launch(self, executable):
|
|
|
|
basedir = self.mktemp()
|
|
|
|
os.mkdir(basedir)
|
|
|
|
private_dir = os.path.join(basedir, "private")
|
|
|
|
os.mkdir(private_dir)
|
|
|
|
reactor = object()
|
|
|
|
args = ["--listen=tor", "--tor-launch"]
|
|
|
|
if executable:
|
|
|
|
args.append("--tor-executable=%s" % executable)
|
|
|
|
cli_config = make_cli_config(basedir, *args)
|
|
|
|
protocol = object()
|
|
|
|
launch_tor = mock.Mock(return_value=defer.succeed(("control_endpoint",
|
|
|
|
protocol)))
|
|
|
|
txtorcon = mock.Mock()
|
|
|
|
ehs = mock.Mock()
|
|
|
|
ehs.private_key = "privkey"
|
|
|
|
ehs.hostname = "ONION.onion"
|
|
|
|
txtorcon.EphemeralHiddenService = mock.Mock(return_value=ehs)
|
|
|
|
ehs.add_to_tor = mock.Mock(return_value=defer.succeed(None))
|
|
|
|
ehs.remove_from_tor = mock.Mock(return_value=defer.succeed(None))
|
|
|
|
|
|
|
|
with mock_txtorcon(txtorcon):
|
|
|
|
with mock.patch("allmydata.util.tor_provider._launch_tor",
|
|
|
|
launch_tor):
|
|
|
|
with mock.patch("allmydata.util.tor_provider.allocate_tcp_port",
|
|
|
|
return_value=999999):
|
2017-07-19 17:09:48 +00:00
|
|
|
d = tor_provider.create_config(reactor, cli_config)
|
2016-10-09 04:57:25 +00:00
|
|
|
tahoe_config_tor, tor_port, tor_location = self.successResultOf(d)
|
|
|
|
|
|
|
|
launch_tor.assert_called_with(reactor, executable,
|
|
|
|
os.path.abspath(private_dir), txtorcon)
|
2016-10-09 00:34:46 +00:00
|
|
|
txtorcon.EphemeralHiddenService.assert_called_with("3457 127.0.0.1:999999")
|
2016-10-09 04:57:25 +00:00
|
|
|
ehs.add_to_tor.assert_called_with(protocol)
|
|
|
|
ehs.remove_from_tor.assert_called_with(protocol)
|
|
|
|
|
|
|
|
expected = {"launch": "true",
|
|
|
|
"onion": "true",
|
|
|
|
"onion.local_port": "999999",
|
|
|
|
"onion.external_port": "3457",
|
2016-10-09 05:49:32 +00:00
|
|
|
"onion.private_key_file": os.path.join("private",
|
|
|
|
"tor_onion.privkey"),
|
2016-10-09 04:57:25 +00:00
|
|
|
}
|
|
|
|
if executable:
|
|
|
|
expected["tor.executable"] = executable
|
|
|
|
self.assertEqual(tahoe_config_tor, expected)
|
|
|
|
self.assertEqual(tor_port, "tcp:999999:interface=127.0.0.1")
|
|
|
|
self.assertEqual(tor_location, "tor:ONION.onion:3457")
|
|
|
|
fn = os.path.join(basedir, tahoe_config_tor["onion.private_key_file"])
|
|
|
|
with open(fn, "rb") as f:
|
|
|
|
privkey = f.read()
|
|
|
|
self.assertEqual(privkey, "privkey")
|
|
|
|
|
|
|
|
def test_launch(self):
|
|
|
|
return self._do_test_launch(None)
|
|
|
|
def test_launch_executable(self):
|
|
|
|
return self._do_test_launch("mytor")
|
|
|
|
|
|
|
|
def test_control_endpoint(self):
|
|
|
|
basedir = self.mktemp()
|
|
|
|
os.mkdir(basedir)
|
|
|
|
private_dir = os.path.join(basedir, "private")
|
|
|
|
os.mkdir(private_dir)
|
|
|
|
reactor = object()
|
|
|
|
cli_config = make_cli_config(basedir, "--listen=tor")
|
|
|
|
protocol = object()
|
|
|
|
connect_to_tor = mock.Mock(return_value=defer.succeed(("goodport",
|
|
|
|
protocol)))
|
|
|
|
txtorcon = mock.Mock()
|
|
|
|
ehs = mock.Mock()
|
|
|
|
ehs.private_key = "privkey"
|
|
|
|
ehs.hostname = "ONION.onion"
|
|
|
|
txtorcon.EphemeralHiddenService = mock.Mock(return_value=ehs)
|
|
|
|
ehs.add_to_tor = mock.Mock(return_value=defer.succeed(None))
|
|
|
|
ehs.remove_from_tor = mock.Mock(return_value=defer.succeed(None))
|
|
|
|
|
|
|
|
with mock_txtorcon(txtorcon):
|
|
|
|
with mock.patch("allmydata.util.tor_provider._connect_to_tor",
|
|
|
|
connect_to_tor):
|
|
|
|
with mock.patch("allmydata.util.tor_provider.allocate_tcp_port",
|
|
|
|
return_value=999999):
|
2017-07-19 17:09:48 +00:00
|
|
|
d = tor_provider.create_config(reactor, cli_config)
|
2016-10-09 04:57:25 +00:00
|
|
|
tahoe_config_tor, tor_port, tor_location = self.successResultOf(d)
|
|
|
|
|
|
|
|
connect_to_tor.assert_called_with(reactor, cli_config, txtorcon)
|
2016-10-09 00:34:46 +00:00
|
|
|
txtorcon.EphemeralHiddenService.assert_called_with("3457 127.0.0.1:999999")
|
2016-10-09 04:57:25 +00:00
|
|
|
ehs.add_to_tor.assert_called_with(protocol)
|
|
|
|
ehs.remove_from_tor.assert_called_with(protocol)
|
|
|
|
|
|
|
|
expected = {"control.port": "goodport",
|
|
|
|
"onion": "true",
|
|
|
|
"onion.local_port": "999999",
|
|
|
|
"onion.external_port": "3457",
|
2016-10-09 05:49:32 +00:00
|
|
|
"onion.private_key_file": os.path.join("private",
|
|
|
|
"tor_onion.privkey"),
|
2016-10-09 04:57:25 +00:00
|
|
|
}
|
|
|
|
self.assertEqual(tahoe_config_tor, expected)
|
|
|
|
self.assertEqual(tor_port, "tcp:999999:interface=127.0.0.1")
|
|
|
|
self.assertEqual(tor_location, "tor:ONION.onion:3457")
|
|
|
|
fn = os.path.join(basedir, tahoe_config_tor["onion.private_key_file"])
|
|
|
|
with open(fn, "rb") as f:
|
|
|
|
privkey = f.read()
|
|
|
|
self.assertEqual(privkey, "privkey")
|
|
|
|
|
2018-01-31 18:30:46 +00:00
|
|
|
|
2016-10-09 04:57:25 +00:00
|
|
|
_None = object()
|
|
|
|
class FakeConfig(dict):
|
|
|
|
def get_config(self, section, option, default=_None, boolean=False):
|
|
|
|
if section != "tor":
|
|
|
|
raise ValueError(section)
|
|
|
|
value = self.get(option, default)
|
|
|
|
if value is _None:
|
|
|
|
raise KeyError
|
|
|
|
return value
|
|
|
|
|
2018-01-31 18:30:46 +00:00
|
|
|
def get_config_path(self, *args):
|
|
|
|
return os.path.join(self.get("basedir", "basedir"), *args)
|
|
|
|
|
|
|
|
|
2016-12-08 23:11:45 +00:00
|
|
|
class EmptyContext(object):
|
|
|
|
def __init__(self):
|
|
|
|
pass
|
|
|
|
def __enter__(self):
|
|
|
|
pass
|
|
|
|
def __exit__(self, type, value, traceback):
|
|
|
|
pass
|
|
|
|
|
2016-10-09 04:57:25 +00:00
|
|
|
class Provider(unittest.TestCase):
|
|
|
|
def test_build(self):
|
2018-01-31 18:30:46 +00:00
|
|
|
tor_provider.Provider(FakeConfig(), "reactor")
|
2016-10-09 04:57:25 +00:00
|
|
|
|
|
|
|
def test_handler_disabled(self):
|
2018-01-31 18:30:46 +00:00
|
|
|
p = tor_provider.Provider(FakeConfig(enabled=False),
|
2016-10-09 04:57:25 +00:00
|
|
|
"reactor")
|
|
|
|
self.assertEqual(p.get_tor_handler(), None)
|
|
|
|
|
|
|
|
def test_handler_no_tor(self):
|
|
|
|
with mock_tor(None):
|
2018-01-31 18:30:46 +00:00
|
|
|
p = tor_provider.Provider(FakeConfig(), "reactor")
|
2016-10-09 04:57:25 +00:00
|
|
|
self.assertEqual(p.get_tor_handler(), None)
|
|
|
|
|
|
|
|
def test_handler_launch_no_txtorcon(self):
|
|
|
|
with mock_txtorcon(None):
|
2018-01-31 18:30:46 +00:00
|
|
|
p = tor_provider.Provider(FakeConfig(launch=True),
|
2016-10-09 04:57:25 +00:00
|
|
|
"reactor")
|
|
|
|
self.assertEqual(p.get_tor_handler(), None)
|
|
|
|
|
|
|
|
@defer.inlineCallbacks
|
|
|
|
def test_handler_launch(self):
|
|
|
|
reactor = object()
|
|
|
|
tor = mock.Mock()
|
|
|
|
txtorcon = mock.Mock()
|
|
|
|
handler = object()
|
|
|
|
tor.control_endpoint_maker = mock.Mock(return_value=handler)
|
2016-12-08 23:11:45 +00:00
|
|
|
tor.add_context = mock.Mock(return_value=EmptyContext())
|
2016-10-09 04:57:25 +00:00
|
|
|
with mock_tor(tor):
|
|
|
|
with mock_txtorcon(txtorcon):
|
2018-01-31 18:30:46 +00:00
|
|
|
p = tor_provider.Provider(FakeConfig(launch=True),
|
2016-10-09 04:57:25 +00:00
|
|
|
reactor)
|
|
|
|
h = p.get_tor_handler()
|
|
|
|
self.assertIs(h, handler)
|
2016-12-08 23:11:45 +00:00
|
|
|
tor.control_endpoint_maker.assert_called_with(p._make_control_endpoint,
|
|
|
|
takes_status=True)
|
2016-10-09 04:57:25 +00:00
|
|
|
|
|
|
|
# make sure Tor is launched just once, the first time an endpoint is
|
|
|
|
# requested, and never again. The clientFromString() function is
|
|
|
|
# called once each time.
|
|
|
|
|
|
|
|
ep_desc = object()
|
|
|
|
launch_tor = mock.Mock(return_value=defer.succeed((ep_desc,None)))
|
|
|
|
ep = object()
|
|
|
|
cfs = mock.Mock(return_value=ep)
|
|
|
|
with mock.patch("allmydata.util.tor_provider._launch_tor", launch_tor):
|
|
|
|
with mock.patch("allmydata.util.tor_provider.clientFromString", cfs):
|
2016-12-08 23:11:45 +00:00
|
|
|
d = p._make_control_endpoint(reactor,
|
|
|
|
update_status=lambda status: None)
|
2016-10-09 04:57:25 +00:00
|
|
|
yield flushEventualQueue()
|
|
|
|
self.assertIs(self.successResultOf(d), ep)
|
2016-10-09 05:49:32 +00:00
|
|
|
launch_tor.assert_called_with(reactor, None,
|
|
|
|
os.path.join("basedir", "private"),
|
2016-10-09 04:57:25 +00:00
|
|
|
txtorcon)
|
|
|
|
cfs.assert_called_with(reactor, ep_desc)
|
|
|
|
|
|
|
|
launch_tor2 = mock.Mock(return_value=defer.succeed((ep_desc,None)))
|
|
|
|
cfs2 = mock.Mock(return_value=ep)
|
|
|
|
with mock.patch("allmydata.util.tor_provider._launch_tor", launch_tor2):
|
|
|
|
with mock.patch("allmydata.util.tor_provider.clientFromString", cfs2):
|
2016-12-08 23:11:45 +00:00
|
|
|
d2 = p._make_control_endpoint(reactor,
|
|
|
|
update_status=lambda status: None)
|
2016-10-09 04:57:25 +00:00
|
|
|
yield flushEventualQueue()
|
|
|
|
self.assertIs(self.successResultOf(d2), ep)
|
|
|
|
self.assertEqual(launch_tor2.mock_calls, [])
|
|
|
|
cfs2.assert_called_with(reactor, ep_desc)
|
|
|
|
|
|
|
|
def test_handler_socks_endpoint(self):
|
|
|
|
tor = mock.Mock()
|
|
|
|
handler = object()
|
|
|
|
tor.socks_endpoint = mock.Mock(return_value=handler)
|
|
|
|
ep = object()
|
|
|
|
cfs = mock.Mock(return_value=ep)
|
|
|
|
reactor = object()
|
|
|
|
|
|
|
|
with mock_tor(tor):
|
2018-01-31 18:30:46 +00:00
|
|
|
p = tor_provider.Provider(FakeConfig(**{"socks.port": "ep_desc"}),
|
2016-10-09 04:57:25 +00:00
|
|
|
reactor)
|
|
|
|
with mock.patch("allmydata.util.tor_provider.clientFromString", cfs):
|
|
|
|
h = p.get_tor_handler()
|
|
|
|
cfs.assert_called_with(reactor, "ep_desc")
|
|
|
|
tor.socks_endpoint.assert_called_with(ep)
|
|
|
|
self.assertIs(h, handler)
|
|
|
|
|
|
|
|
def test_handler_control_endpoint(self):
|
|
|
|
tor = mock.Mock()
|
|
|
|
handler = object()
|
|
|
|
tor.control_endpoint = mock.Mock(return_value=handler)
|
|
|
|
ep = object()
|
|
|
|
cfs = mock.Mock(return_value=ep)
|
|
|
|
reactor = object()
|
|
|
|
|
|
|
|
with mock_tor(tor):
|
2018-01-31 18:30:46 +00:00
|
|
|
p = tor_provider.Provider(FakeConfig(**{"control.port": "ep_desc"}),
|
2016-10-09 04:57:25 +00:00
|
|
|
reactor)
|
|
|
|
with mock.patch("allmydata.util.tor_provider.clientFromString", cfs):
|
|
|
|
h = p.get_tor_handler()
|
|
|
|
self.assertIs(h, handler)
|
|
|
|
cfs.assert_called_with(reactor, "ep_desc")
|
|
|
|
tor.control_endpoint.assert_called_with(ep)
|
|
|
|
|
|
|
|
def test_handler_default(self):
|
|
|
|
tor = mock.Mock()
|
|
|
|
handler = object()
|
|
|
|
tor.default_socks = mock.Mock(return_value=handler)
|
|
|
|
|
|
|
|
with mock_tor(tor):
|
2018-01-31 18:30:46 +00:00
|
|
|
p = tor_provider.Provider(FakeConfig(), "reactor")
|
2016-10-09 04:57:25 +00:00
|
|
|
h = p.get_tor_handler()
|
|
|
|
self.assertIs(h, handler)
|
|
|
|
tor.default_socks.assert_called_with()
|
|
|
|
|
2017-08-16 00:59:59 +00:00
|
|
|
class ProviderListener(unittest.TestCase):
|
2017-07-19 17:24:24 +00:00
|
|
|
def test_listener(self):
|
2017-08-16 00:59:59 +00:00
|
|
|
"""Does the Tor Provider object's get_listener() method correctly
|
|
|
|
convert the [tor] section of tahoe.cfg into an
|
|
|
|
endpoint/descriptor?
|
|
|
|
"""
|
2017-07-19 17:24:24 +00:00
|
|
|
tor = mock.Mock()
|
|
|
|
handler = object()
|
|
|
|
tor.socks_endpoint = mock.Mock(return_value=handler)
|
|
|
|
reactor = object()
|
|
|
|
|
|
|
|
with mock_tor(tor):
|
2018-01-31 18:30:46 +00:00
|
|
|
p = tor_provider.Provider(FakeConfig(**{"onion.local_port": "321"}),
|
2017-07-19 17:24:24 +00:00
|
|
|
reactor)
|
2017-08-16 01:00:24 +00:00
|
|
|
fake_ep = object()
|
|
|
|
with mock.patch("allmydata.util.tor_provider.TCP4ServerEndpoint",
|
|
|
|
return_value=fake_ep) as e:
|
2017-07-19 17:24:24 +00:00
|
|
|
endpoint_or_description = p.get_listener()
|
2017-08-16 01:00:24 +00:00
|
|
|
self.assertIs(endpoint_or_description, fake_ep)
|
|
|
|
self.assertEqual(e.mock_calls, [mock.call(reactor, 321,
|
|
|
|
interface="127.0.0.1")])
|
2017-07-19 17:24:24 +00:00
|
|
|
|
2016-10-09 04:57:25 +00:00
|
|
|
class Provider_CheckOnionConfig(unittest.TestCase):
|
|
|
|
def test_default(self):
|
|
|
|
# default config doesn't start an onion service, so it should be
|
|
|
|
# happy both with and without txtorcon
|
|
|
|
|
2018-01-31 18:30:46 +00:00
|
|
|
p = tor_provider.Provider(FakeConfig(), "reactor")
|
2016-10-09 04:57:25 +00:00
|
|
|
p.check_onion_config()
|
|
|
|
|
|
|
|
with mock_txtorcon(None):
|
2018-01-31 18:30:46 +00:00
|
|
|
p = tor_provider.Provider(FakeConfig(), "reactor")
|
2016-10-09 04:57:25 +00:00
|
|
|
p.check_onion_config()
|
|
|
|
|
|
|
|
def test_no_txtorcon(self):
|
|
|
|
with mock_txtorcon(None):
|
2018-01-31 18:30:46 +00:00
|
|
|
p = tor_provider.Provider(FakeConfig(onion=True),
|
2016-10-09 04:57:25 +00:00
|
|
|
"reactor")
|
|
|
|
e = self.assertRaises(ValueError, p.check_onion_config)
|
|
|
|
self.assertEqual(str(e), "Cannot create onion without txtorcon. "
|
|
|
|
"Please 'pip install tahoe-lafs[tor]' to fix.")
|
|
|
|
|
|
|
|
def test_no_launch_no_control(self):
|
2018-01-31 18:30:46 +00:00
|
|
|
p = tor_provider.Provider(FakeConfig(onion=True), "reactor")
|
2016-10-09 04:57:25 +00:00
|
|
|
e = self.assertRaises(ValueError, p.check_onion_config)
|
|
|
|
self.assertEqual(str(e), "[tor] onion = true, but we have neither "
|
|
|
|
"launch=true nor control.port=")
|
|
|
|
|
|
|
|
def test_missing_keys(self):
|
2018-01-31 18:30:46 +00:00
|
|
|
p = tor_provider.Provider(FakeConfig(onion=True,
|
|
|
|
launch=True), "reactor")
|
2016-10-09 04:57:25 +00:00
|
|
|
e = self.assertRaises(ValueError, p.check_onion_config)
|
|
|
|
self.assertEqual(str(e), "[tor] onion = true, "
|
|
|
|
"but onion.local_port= is missing")
|
|
|
|
|
2018-01-31 18:30:46 +00:00
|
|
|
p = tor_provider.Provider(FakeConfig(onion=True, launch=True,
|
2016-10-09 04:57:25 +00:00
|
|
|
**{"onion.local_port": "x",
|
|
|
|
}), "reactor")
|
|
|
|
e = self.assertRaises(ValueError, p.check_onion_config)
|
|
|
|
self.assertEqual(str(e), "[tor] onion = true, "
|
|
|
|
"but onion.external_port= is missing")
|
|
|
|
|
2018-01-31 18:30:46 +00:00
|
|
|
p = tor_provider.Provider(FakeConfig(onion=True, launch=True,
|
2016-10-09 04:57:25 +00:00
|
|
|
**{"onion.local_port": "x",
|
|
|
|
"onion.external_port": "y",
|
|
|
|
}), "reactor")
|
|
|
|
e = self.assertRaises(ValueError, p.check_onion_config)
|
|
|
|
self.assertEqual(str(e), "[tor] onion = true, "
|
|
|
|
"but onion.private_key_file= is missing")
|
|
|
|
|
|
|
|
def test_ok(self):
|
2018-01-31 18:30:46 +00:00
|
|
|
p = tor_provider.Provider(FakeConfig(onion=True, launch=True,
|
2016-10-09 04:57:25 +00:00
|
|
|
**{"onion.local_port": "x",
|
|
|
|
"onion.external_port": "y",
|
|
|
|
"onion.private_key_file": "z",
|
|
|
|
}), "reactor")
|
|
|
|
p.check_onion_config()
|
|
|
|
|
|
|
|
class Provider_Service(unittest.TestCase):
|
|
|
|
def test_no_onion(self):
|
|
|
|
reactor = object()
|
2018-01-31 18:30:46 +00:00
|
|
|
p = tor_provider.Provider(FakeConfig(onion=False), reactor)
|
2016-10-09 04:57:25 +00:00
|
|
|
with mock.patch("allmydata.util.tor_provider.Provider._start_onion") as s:
|
|
|
|
p.startService()
|
|
|
|
self.assertEqual(s.mock_calls, [])
|
|
|
|
self.assertEqual(p.running, True)
|
|
|
|
|
|
|
|
p.stopService()
|
|
|
|
self.assertEqual(p.running, False)
|
|
|
|
|
|
|
|
@defer.inlineCallbacks
|
|
|
|
def test_launch(self):
|
|
|
|
basedir = self.mktemp()
|
|
|
|
os.mkdir(basedir)
|
|
|
|
fn = os.path.join(basedir, "keyfile")
|
|
|
|
with open(fn, "w") as f:
|
|
|
|
f.write("private key")
|
|
|
|
reactor = object()
|
2018-01-31 18:30:46 +00:00
|
|
|
cfg = FakeConfig(basedir=basedir, onion=True, launch=True,
|
2016-10-09 04:57:25 +00:00
|
|
|
**{"onion.local_port": 123,
|
|
|
|
"onion.external_port": 456,
|
|
|
|
"onion.private_key_file": "keyfile",
|
|
|
|
})
|
|
|
|
|
|
|
|
txtorcon = mock.Mock()
|
|
|
|
with mock_txtorcon(txtorcon):
|
2018-01-31 18:30:46 +00:00
|
|
|
p = tor_provider.Provider(cfg, reactor)
|
2016-10-09 04:57:25 +00:00
|
|
|
tor_state = mock.Mock()
|
|
|
|
tor_state.protocol = object()
|
|
|
|
ehs = mock.Mock()
|
|
|
|
ehs.add_to_tor = mock.Mock(return_value=defer.succeed(None))
|
|
|
|
ehs.remove_from_tor = mock.Mock(return_value=defer.succeed(None))
|
|
|
|
txtorcon.EphemeralHiddenService = mock.Mock(return_value=ehs)
|
|
|
|
launch_tor = mock.Mock(return_value=defer.succeed((None,tor_state.protocol)))
|
|
|
|
with mock.patch("allmydata.util.tor_provider._launch_tor",
|
|
|
|
launch_tor):
|
|
|
|
d = p.startService()
|
|
|
|
yield flushEventualQueue()
|
|
|
|
self.successResultOf(d)
|
|
|
|
self.assertIs(p._onion_ehs, ehs)
|
|
|
|
self.assertIs(p._onion_tor_control_proto, tor_state.protocol)
|
|
|
|
launch_tor.assert_called_with(reactor, None,
|
|
|
|
os.path.join(basedir, "private"), txtorcon)
|
2016-10-09 00:34:46 +00:00
|
|
|
txtorcon.EphemeralHiddenService.assert_called_with("456 127.0.0.1:123",
|
2016-10-09 04:57:25 +00:00
|
|
|
"private key")
|
|
|
|
ehs.add_to_tor.assert_called_with(tor_state.protocol)
|
|
|
|
|
|
|
|
yield p.stopService()
|
|
|
|
ehs.remove_from_tor.assert_called_with(tor_state.protocol)
|
|
|
|
|
|
|
|
@defer.inlineCallbacks
|
|
|
|
def test_control_endpoint(self):
|
|
|
|
basedir = self.mktemp()
|
|
|
|
os.mkdir(basedir)
|
|
|
|
fn = os.path.join(basedir, "keyfile")
|
|
|
|
with open(fn, "w") as f:
|
|
|
|
f.write("private key")
|
|
|
|
reactor = object()
|
2018-01-31 18:30:46 +00:00
|
|
|
cfg = FakeConfig(basedir=basedir, onion=True,
|
2016-10-09 04:57:25 +00:00
|
|
|
**{"control.port": "ep_desc",
|
|
|
|
"onion.local_port": 123,
|
|
|
|
"onion.external_port": 456,
|
|
|
|
"onion.private_key_file": "keyfile",
|
|
|
|
})
|
|
|
|
|
|
|
|
txtorcon = mock.Mock()
|
|
|
|
with mock_txtorcon(txtorcon):
|
2018-01-31 18:30:46 +00:00
|
|
|
p = tor_provider.Provider(cfg, reactor)
|
2016-10-09 04:57:25 +00:00
|
|
|
tor_state = mock.Mock()
|
|
|
|
tor_state.protocol = object()
|
|
|
|
txtorcon.build_tor_connection = mock.Mock(return_value=tor_state)
|
|
|
|
ehs = mock.Mock()
|
|
|
|
ehs.add_to_tor = mock.Mock(return_value=defer.succeed(None))
|
|
|
|
ehs.remove_from_tor = mock.Mock(return_value=defer.succeed(None))
|
|
|
|
txtorcon.EphemeralHiddenService = mock.Mock(return_value=ehs)
|
|
|
|
tcep = object()
|
|
|
|
cfs = mock.Mock(return_value=tcep)
|
|
|
|
with mock.patch("allmydata.util.tor_provider.clientFromString", cfs):
|
|
|
|
d = p.startService()
|
|
|
|
yield flushEventualQueue()
|
|
|
|
self.successResultOf(d)
|
|
|
|
self.assertIs(p._onion_ehs, ehs)
|
|
|
|
self.assertIs(p._onion_tor_control_proto, tor_state.protocol)
|
|
|
|
cfs.assert_called_with(reactor, "ep_desc")
|
|
|
|
txtorcon.build_tor_connection.assert_called_with(tcep)
|
2016-10-09 00:34:46 +00:00
|
|
|
txtorcon.EphemeralHiddenService.assert_called_with("456 127.0.0.1:123",
|
2016-10-09 04:57:25 +00:00
|
|
|
"private key")
|
|
|
|
ehs.add_to_tor.assert_called_with(tor_state.protocol)
|
|
|
|
|
|
|
|
yield p.stopService()
|
|
|
|
ehs.remove_from_tor.assert_called_with(tor_state.protocol)
|