2016-08-28 09:28:39 +00:00
|
|
|
import os
|
|
|
|
import mock
|
|
|
|
from io import BytesIO
|
|
|
|
from twisted.trial import unittest
|
2016-10-09 05:02:11 +00:00
|
|
|
from twisted.internet import reactor, endpoints, defer
|
2016-09-02 03:17:45 +00:00
|
|
|
from twisted.internet.interfaces import IStreamClientEndpoint
|
2016-08-28 09:28:39 +00:00
|
|
|
from ConfigParser import SafeConfigParser
|
|
|
|
from foolscap.connections import tcp
|
2016-08-31 09:44:27 +00:00
|
|
|
from ..node import Node, PrivacyError
|
2016-08-28 09:28:39 +00:00
|
|
|
|
|
|
|
class FakeNode(Node):
|
|
|
|
def __init__(self, config_str):
|
|
|
|
self.config = SafeConfigParser()
|
|
|
|
self.config.readfp(BytesIO(config_str))
|
2016-08-31 09:44:27 +00:00
|
|
|
self._reveal_ip = True
|
2016-10-09 05:02:11 +00:00
|
|
|
self.basedir = "BASEDIR"
|
|
|
|
self.services = []
|
|
|
|
self.create_tor_provider()
|
2016-08-28 09:28:39 +00:00
|
|
|
|
|
|
|
BASECONFIG = ("[client]\n"
|
|
|
|
"introducer.furl = \n"
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
class TCP(unittest.TestCase):
|
|
|
|
def test_default(self):
|
|
|
|
n = FakeNode(BASECONFIG)
|
|
|
|
h = n._make_tcp_handler()
|
|
|
|
self.assertIsInstance(h, tcp.DefaultTCP)
|
|
|
|
|
|
|
|
class Tor(unittest.TestCase):
|
|
|
|
def test_disabled(self):
|
2016-08-31 22:23:47 +00:00
|
|
|
n = FakeNode(BASECONFIG+"[tor]\nenabled = false\n")
|
2016-08-28 09:28:39 +00:00
|
|
|
h = n._make_tor_handler()
|
|
|
|
self.assertEqual(h, None)
|
|
|
|
|
2016-08-31 08:50:13 +00:00
|
|
|
def test_unimportable(self):
|
2016-10-09 05:02:11 +00:00
|
|
|
with mock.patch("allmydata.util.tor_provider._import_tor",
|
|
|
|
return_value=None):
|
|
|
|
n = FakeNode(BASECONFIG)
|
2016-08-31 08:50:13 +00:00
|
|
|
h = n._make_tor_handler()
|
|
|
|
self.assertEqual(h, None)
|
|
|
|
|
2016-08-28 09:28:39 +00:00
|
|
|
def test_default(self):
|
|
|
|
h1 = mock.Mock()
|
|
|
|
with mock.patch("foolscap.connections.tor.default_socks",
|
|
|
|
return_value=h1) as f:
|
2016-10-09 05:02:11 +00:00
|
|
|
n = FakeNode(BASECONFIG)
|
2016-08-28 09:28:39 +00:00
|
|
|
h = n._make_tor_handler()
|
|
|
|
self.assertEqual(f.mock_calls, [mock.call()])
|
|
|
|
self.assertIdentical(h, h1)
|
|
|
|
|
2016-10-09 05:02:11 +00:00
|
|
|
def _do_test_launch(self, executable):
|
|
|
|
# the handler is created right away
|
|
|
|
config = BASECONFIG+"[tor]\nlaunch = true\n"
|
|
|
|
if executable:
|
|
|
|
config += "tor.executable = %s\n" % executable
|
2016-08-28 09:28:39 +00:00
|
|
|
h1 = mock.Mock()
|
2016-10-09 05:02:11 +00:00
|
|
|
with mock.patch("foolscap.connections.tor.control_endpoint_maker",
|
2016-08-28 09:28:39 +00:00
|
|
|
return_value=h1) as f:
|
2016-10-09 05:02:11 +00:00
|
|
|
n = FakeNode(config)
|
2016-08-28 09:28:39 +00:00
|
|
|
h = n._make_tor_handler()
|
2016-10-09 05:02:11 +00:00
|
|
|
private_dir = os.path.join(n.basedir, "private")
|
|
|
|
exp = mock.call(n._tor_provider._make_control_endpoint)
|
2016-08-28 09:28:39 +00:00
|
|
|
self.assertEqual(f.mock_calls, [exp])
|
|
|
|
self.assertIdentical(h, h1)
|
|
|
|
|
2016-10-09 05:02:11 +00:00
|
|
|
# later, when Foolscap first connects, Tor should be launched
|
|
|
|
tp = n._tor_provider
|
|
|
|
reactor = "reactor"
|
|
|
|
tcp = object()
|
|
|
|
tcep = object()
|
|
|
|
launch_tor = mock.Mock(return_value=defer.succeed(("ep_desc", tcp)))
|
|
|
|
cfs = mock.Mock(return_value=tcep)
|
|
|
|
with mock.patch("allmydata.util.tor_provider._launch_tor", launch_tor):
|
|
|
|
with mock.patch("allmydata.util.tor_provider.clientFromString", cfs):
|
|
|
|
cep = self.successResultOf(tp._make_control_endpoint(reactor))
|
|
|
|
launch_tor.assert_called_with(reactor, executable, private_dir,
|
|
|
|
tp._txtorcon)
|
|
|
|
cfs.assert_called_with(reactor, "ep_desc")
|
|
|
|
self.assertIs(cep, tcep)
|
|
|
|
|
|
|
|
def test_launch(self):
|
|
|
|
self._do_test_launch(None)
|
|
|
|
|
2016-08-28 09:28:39 +00:00
|
|
|
def test_launch_executable(self):
|
2016-10-09 05:02:11 +00:00
|
|
|
self._do_test_launch("/special/tor")
|
2016-08-28 09:28:39 +00:00
|
|
|
|
2016-09-02 03:17:45 +00:00
|
|
|
def test_socksport_unix_endpoint(self):
|
|
|
|
h1 = mock.Mock()
|
|
|
|
with mock.patch("foolscap.connections.tor.socks_endpoint",
|
|
|
|
return_value=h1) as f:
|
2016-10-09 05:02:11 +00:00
|
|
|
n = FakeNode(BASECONFIG+"[tor]\nsocks.port = unix:/var/lib/fw-daemon/tor_socks.socket\n")
|
2016-09-02 03:17:45 +00:00
|
|
|
h = n._make_tor_handler()
|
|
|
|
self.assertTrue(IStreamClientEndpoint.providedBy(f.mock_calls[0]))
|
|
|
|
self.assertIdentical(h, h1)
|
|
|
|
|
|
|
|
def test_socksport_endpoint(self):
|
2016-08-28 09:28:39 +00:00
|
|
|
h1 = mock.Mock()
|
2016-09-02 03:17:45 +00:00
|
|
|
with mock.patch("foolscap.connections.tor.socks_endpoint",
|
2016-08-28 09:28:39 +00:00
|
|
|
return_value=h1) as f:
|
2016-10-09 05:02:11 +00:00
|
|
|
n = FakeNode(BASECONFIG+"[tor]\nsocks.port = tcp:127.0.0.1:1234\n")
|
2016-08-28 09:28:39 +00:00
|
|
|
h = n._make_tor_handler()
|
2016-09-02 03:17:45 +00:00
|
|
|
self.assertTrue(IStreamClientEndpoint.providedBy(f.mock_calls[0]))
|
2016-08-28 09:28:39 +00:00
|
|
|
self.assertIdentical(h, h1)
|
|
|
|
|
2016-09-02 03:17:45 +00:00
|
|
|
def test_socksport_endpoint_otherhost(self):
|
2016-08-28 09:28:39 +00:00
|
|
|
h1 = mock.Mock()
|
2016-09-02 03:17:45 +00:00
|
|
|
with mock.patch("foolscap.connections.tor.socks_endpoint",
|
2016-08-28 09:28:39 +00:00
|
|
|
return_value=h1) as f:
|
2016-10-09 05:02:11 +00:00
|
|
|
n = FakeNode(BASECONFIG+"[tor]\nsocks.port = tcp:otherhost:1234\n")
|
2016-08-28 09:28:39 +00:00
|
|
|
h = n._make_tor_handler()
|
2016-09-02 03:17:45 +00:00
|
|
|
self.assertTrue(IStreamClientEndpoint.providedBy(f.mock_calls[0]))
|
2016-08-28 09:28:39 +00:00
|
|
|
self.assertIdentical(h, h1)
|
|
|
|
|
2016-08-28 23:28:01 +00:00
|
|
|
def test_socksport_bad_endpoint(self):
|
2016-09-02 03:17:45 +00:00
|
|
|
n = FakeNode(BASECONFIG+"[tor]\nsocks.port = meow:unsupported\n")
|
2016-08-28 09:28:39 +00:00
|
|
|
e = self.assertRaises(ValueError, n._make_tor_handler)
|
2016-09-02 03:17:45 +00:00
|
|
|
self.assertIn("Unknown endpoint type: 'meow'", str(e))
|
2016-08-28 09:28:39 +00:00
|
|
|
|
|
|
|
def test_socksport_not_integer(self):
|
2016-08-28 23:28:01 +00:00
|
|
|
n = FakeNode(BASECONFIG+"[tor]\nsocks.port = tcp:localhost:kumquat\n")
|
2016-08-28 09:28:39 +00:00
|
|
|
e = self.assertRaises(ValueError, n._make_tor_handler)
|
2016-09-02 03:17:45 +00:00
|
|
|
self.assertIn("invalid literal for int() with base 10: 'kumquat'", str(e))
|
2016-08-28 09:28:39 +00:00
|
|
|
|
|
|
|
def test_controlport(self):
|
|
|
|
h1 = mock.Mock()
|
|
|
|
with mock.patch("foolscap.connections.tor.control_endpoint",
|
|
|
|
return_value=h1) as f:
|
2016-10-09 05:02:11 +00:00
|
|
|
n = FakeNode(BASECONFIG+"[tor]\ncontrol.port = tcp:localhost:1234\n")
|
2016-08-28 09:28:39 +00:00
|
|
|
h = n._make_tor_handler()
|
|
|
|
self.assertEqual(len(f.mock_calls), 1)
|
|
|
|
ep = f.mock_calls[0][1][0]
|
|
|
|
self.assertIsInstance(ep, endpoints.TCP4ClientEndpoint)
|
|
|
|
self.assertIdentical(h, h1)
|
|
|
|
|
2016-08-28 10:15:35 +00:00
|
|
|
class I2P(unittest.TestCase):
|
|
|
|
def test_disabled(self):
|
2016-08-31 22:23:47 +00:00
|
|
|
n = FakeNode(BASECONFIG+"[i2p]\nenabled = false\n")
|
2016-08-28 10:15:35 +00:00
|
|
|
h = n._make_i2p_handler()
|
|
|
|
self.assertEqual(h, None)
|
|
|
|
|
2016-08-31 08:50:13 +00:00
|
|
|
def test_unimportable(self):
|
|
|
|
n = FakeNode(BASECONFIG)
|
|
|
|
with mock.patch("allmydata.node._import_i2p", return_value=None):
|
|
|
|
h = n._make_i2p_handler()
|
|
|
|
self.assertEqual(h, None)
|
|
|
|
|
2016-08-28 10:15:35 +00:00
|
|
|
def test_default(self):
|
|
|
|
n = FakeNode(BASECONFIG)
|
|
|
|
h1 = mock.Mock()
|
|
|
|
with mock.patch("foolscap.connections.i2p.default",
|
|
|
|
return_value=h1) as f:
|
|
|
|
h = n._make_i2p_handler()
|
|
|
|
self.assertEqual(f.mock_calls, [mock.call(reactor)])
|
|
|
|
self.assertIdentical(h, h1)
|
|
|
|
|
|
|
|
def test_samport(self):
|
|
|
|
n = FakeNode(BASECONFIG+"[i2p]\nsam.port = tcp:localhost:1234\n")
|
|
|
|
h1 = mock.Mock()
|
|
|
|
with mock.patch("foolscap.connections.i2p.sam_endpoint",
|
|
|
|
return_value=h1) as f:
|
|
|
|
h = n._make_i2p_handler()
|
|
|
|
self.assertEqual(len(f.mock_calls), 1)
|
|
|
|
ep = f.mock_calls[0][1][0]
|
|
|
|
self.assertIsInstance(ep, endpoints.TCP4ClientEndpoint)
|
|
|
|
self.assertIdentical(h, h1)
|
|
|
|
|
|
|
|
def test_samport_and_launch(self):
|
|
|
|
n = FakeNode(BASECONFIG+"[i2p]\n" +
|
|
|
|
"sam.port = tcp:localhost:1234\n"
|
|
|
|
+"launch = true\n")
|
|
|
|
e = self.assertRaises(ValueError, n._make_i2p_handler)
|
|
|
|
self.assertIn("must not set both sam.port and launch", str(e))
|
|
|
|
|
|
|
|
def test_launch(self):
|
|
|
|
n = FakeNode(BASECONFIG+"[i2p]\nlaunch = true\n")
|
|
|
|
h1 = mock.Mock()
|
|
|
|
with mock.patch("foolscap.connections.i2p.launch",
|
|
|
|
return_value=h1) as f:
|
|
|
|
h = n._make_i2p_handler()
|
|
|
|
exp = mock.call(i2p_configdir=None, i2p_binary=None)
|
|
|
|
self.assertEqual(f.mock_calls, [exp])
|
|
|
|
self.assertIdentical(h, h1)
|
|
|
|
|
|
|
|
def test_launch_executable(self):
|
|
|
|
n = FakeNode(BASECONFIG+"[i2p]\nlaunch = true\n" +
|
|
|
|
"i2p.executable = i2p\n")
|
|
|
|
h1 = mock.Mock()
|
|
|
|
with mock.patch("foolscap.connections.i2p.launch",
|
|
|
|
return_value=h1) as f:
|
|
|
|
h = n._make_i2p_handler()
|
|
|
|
exp = mock.call(i2p_configdir=None, i2p_binary="i2p")
|
|
|
|
self.assertEqual(f.mock_calls, [exp])
|
|
|
|
self.assertIdentical(h, h1)
|
|
|
|
|
|
|
|
def test_launch_configdir(self):
|
|
|
|
n = FakeNode(BASECONFIG+"[i2p]\nlaunch = true\n" +
|
|
|
|
"i2p.configdir = cfg\n")
|
|
|
|
h1 = mock.Mock()
|
|
|
|
with mock.patch("foolscap.connections.i2p.launch",
|
|
|
|
return_value=h1) as f:
|
|
|
|
h = n._make_i2p_handler()
|
|
|
|
exp = mock.call(i2p_configdir="cfg", i2p_binary=None)
|
|
|
|
self.assertEqual(f.mock_calls, [exp])
|
|
|
|
self.assertIdentical(h, h1)
|
|
|
|
|
|
|
|
def test_launch_configdir_and_executable(self):
|
|
|
|
n = FakeNode(BASECONFIG+"[i2p]\nlaunch = true\n" +
|
|
|
|
"i2p.executable = i2p\n" +
|
|
|
|
"i2p.configdir = cfg\n")
|
|
|
|
h1 = mock.Mock()
|
|
|
|
with mock.patch("foolscap.connections.i2p.launch",
|
|
|
|
return_value=h1) as f:
|
|
|
|
h = n._make_i2p_handler()
|
|
|
|
exp = mock.call(i2p_configdir="cfg", i2p_binary="i2p")
|
|
|
|
self.assertEqual(f.mock_calls, [exp])
|
|
|
|
self.assertIdentical(h, h1)
|
|
|
|
|
|
|
|
def test_configdir(self):
|
|
|
|
n = FakeNode(BASECONFIG+"[i2p]\ni2p.configdir = cfg\n")
|
|
|
|
h1 = mock.Mock()
|
|
|
|
with mock.patch("foolscap.connections.i2p.local_i2p",
|
|
|
|
return_value=h1) as f:
|
|
|
|
h = n._make_i2p_handler()
|
|
|
|
self.assertEqual(f.mock_calls, [mock.call("cfg")])
|
|
|
|
self.assertIdentical(h, h1)
|
|
|
|
|
2016-08-28 09:28:39 +00:00
|
|
|
class Connections(unittest.TestCase):
|
|
|
|
def test_default(self):
|
|
|
|
n = FakeNode(BASECONFIG)
|
|
|
|
n.init_connections()
|
|
|
|
self.assertEqual(n._default_connection_handlers["tcp"], "tcp")
|
|
|
|
self.assertEqual(n._default_connection_handlers["tor"], "tor")
|
|
|
|
self.assertEqual(n._default_connection_handlers["i2p"], "i2p")
|
2016-09-14 23:21:55 +00:00
|
|
|
n.set_tub_options()
|
|
|
|
n._create_tub()
|
2016-08-28 09:28:39 +00:00
|
|
|
|
|
|
|
def test_tor(self):
|
|
|
|
n = FakeNode(BASECONFIG+"[connections]\ntcp = tor\n")
|
|
|
|
n.init_connections()
|
|
|
|
self.assertEqual(n._default_connection_handlers["tcp"], "tor")
|
|
|
|
self.assertEqual(n._default_connection_handlers["tor"], "tor")
|
|
|
|
self.assertEqual(n._default_connection_handlers["i2p"], "i2p")
|
|
|
|
|
2016-08-31 08:50:13 +00:00
|
|
|
def test_tor_unimportable(self):
|
2016-10-09 05:02:11 +00:00
|
|
|
with mock.patch("allmydata.util.tor_provider._import_tor",
|
|
|
|
return_value=None):
|
|
|
|
n = FakeNode(BASECONFIG+"[connections]\ntcp = tor\n")
|
|
|
|
e = self.assertRaises(ValueError, n.init_connections)
|
2016-08-31 08:50:13 +00:00
|
|
|
self.assertEqual(str(e),
|
|
|
|
"'tahoe.cfg [connections] tcp='"
|
|
|
|
" uses unavailable/unimportable handler type 'tor'."
|
|
|
|
" Please pip install tahoe-lafs[tor] to fix.")
|
|
|
|
|
2016-08-28 09:28:39 +00:00
|
|
|
def test_unknown(self):
|
|
|
|
n = FakeNode(BASECONFIG+"[connections]\ntcp = unknown\n")
|
|
|
|
e = self.assertRaises(ValueError, n.init_connections)
|
|
|
|
self.assertIn("'tahoe.cfg [connections] tcp='", str(e))
|
|
|
|
self.assertIn("uses unknown handler type 'unknown'", str(e))
|
2016-08-31 09:44:27 +00:00
|
|
|
|
2016-09-14 23:21:55 +00:00
|
|
|
def test_tcp_disabled(self):
|
|
|
|
n = FakeNode(BASECONFIG+"[connections]\ntcp = disabled\n")
|
|
|
|
n.init_connections()
|
|
|
|
self.assertEqual(n._default_connection_handlers["tcp"], None)
|
|
|
|
self.assertEqual(n._default_connection_handlers["tor"], "tor")
|
|
|
|
self.assertEqual(n._default_connection_handlers["i2p"], "i2p")
|
|
|
|
n.set_tub_options()
|
|
|
|
n._create_tub()
|
|
|
|
|
2016-08-31 09:44:27 +00:00
|
|
|
class Privacy(unittest.TestCase):
|
|
|
|
def test_flag(self):
|
|
|
|
n = FakeNode(BASECONFIG)
|
|
|
|
n.check_privacy()
|
|
|
|
self.assertTrue(n._reveal_ip)
|
|
|
|
|
|
|
|
n = FakeNode(BASECONFIG+"[node]\nreveal-IP-address = true\n")
|
|
|
|
n.check_privacy()
|
|
|
|
self.assertTrue(n._reveal_ip)
|
|
|
|
|
|
|
|
n = FakeNode(BASECONFIG+"[node]\nreveal-IP-address = false\n")
|
|
|
|
n.check_privacy()
|
|
|
|
self.assertFalse(n._reveal_ip)
|
|
|
|
|
|
|
|
n = FakeNode(BASECONFIG+"[node]\nreveal-ip-address = false\n")
|
|
|
|
n.check_privacy()
|
|
|
|
self.assertFalse(n._reveal_ip)
|
|
|
|
|
|
|
|
def test_connections(self):
|
|
|
|
n = FakeNode(BASECONFIG+"[node]\nreveal-IP-address = false\n")
|
|
|
|
n.check_privacy()
|
|
|
|
e = self.assertRaises(PrivacyError, n.init_connections)
|
2016-09-20 17:14:35 +00:00
|
|
|
self.assertEqual(str(e),
|
|
|
|
"tcp = tcp, must be set to 'tor' or 'disabled'")
|
2016-08-31 09:44:27 +00:00
|
|
|
|
2016-09-14 23:21:55 +00:00
|
|
|
def test_connections_tcp_disabled(self):
|
|
|
|
n = FakeNode(BASECONFIG+
|
|
|
|
"[connections]\ntcp = disabled\n"+
|
|
|
|
"[node]\nreveal-IP-address = false\n")
|
|
|
|
n.check_privacy()
|
|
|
|
n.init_connections() # passes privacy check
|
|
|
|
self.assertEqual(n._default_connection_handlers["tcp"], None)
|
|
|
|
|
2016-08-31 09:44:27 +00:00
|
|
|
def test_tub_location_auto(self):
|
|
|
|
n = FakeNode(BASECONFIG+"[node]\nreveal-IP-address = false\n")
|
|
|
|
n._portnumfile = "missing"
|
|
|
|
n.check_privacy()
|
2016-09-01 00:16:23 +00:00
|
|
|
e = self.assertRaises(PrivacyError, n.get_tub_portlocation, None, None)
|
2016-08-31 09:44:27 +00:00
|
|
|
self.assertEqual(str(e), "tub.location uses AUTO")
|
|
|
|
|
2016-09-01 00:16:23 +00:00
|
|
|
n = FakeNode(BASECONFIG+"[node]\nreveal-IP-address = false\n")
|
2016-08-31 09:44:27 +00:00
|
|
|
n._portnumfile = "missing"
|
|
|
|
n.check_privacy()
|
2016-09-01 00:16:23 +00:00
|
|
|
e = self.assertRaises(PrivacyError, n.get_tub_portlocation,
|
|
|
|
None, "AUTO")
|
2016-08-31 09:44:27 +00:00
|
|
|
self.assertEqual(str(e), "tub.location uses AUTO")
|
|
|
|
|
2016-09-01 00:16:23 +00:00
|
|
|
n = FakeNode(BASECONFIG+"[node]\nreveal-IP-address = false\n")
|
2016-08-31 09:44:27 +00:00
|
|
|
n._portnumfile = "missing"
|
|
|
|
n.check_privacy()
|
2016-09-01 00:16:23 +00:00
|
|
|
e = self.assertRaises(PrivacyError, n.get_tub_portlocation,
|
|
|
|
None, "AUTO,tcp:hostname:1234")
|
2016-08-31 09:44:27 +00:00
|
|
|
self.assertEqual(str(e), "tub.location uses AUTO")
|
|
|
|
|
|
|
|
def test_tub_location_tcp(self):
|
2016-09-01 00:16:23 +00:00
|
|
|
n = FakeNode(BASECONFIG+"[node]\nreveal-IP-address = false\n")
|
2016-08-31 09:44:27 +00:00
|
|
|
n._portnumfile = "missing"
|
|
|
|
n.check_privacy()
|
2016-09-01 00:16:23 +00:00
|
|
|
e = self.assertRaises(PrivacyError, n.get_tub_portlocation,
|
|
|
|
None, "tcp:hostname:1234")
|
2016-08-31 09:44:27 +00:00
|
|
|
self.assertEqual(str(e), "tub.location includes tcp: hint")
|
|
|
|
|
2016-09-02 16:25:26 +00:00
|
|
|
def test_tub_location_legacy_tcp(self):
|
|
|
|
n = FakeNode(BASECONFIG+"[node]\nreveal-IP-address = false\n")
|
|
|
|
n._portnumfile = "missing"
|
|
|
|
n.check_privacy()
|
|
|
|
e = self.assertRaises(PrivacyError, n.get_tub_portlocation,
|
|
|
|
None, "hostname:1234")
|
|
|
|
self.assertEqual(str(e), "tub.location includes tcp: hint")
|
|
|
|
|
2016-08-31 09:44:27 +00:00
|
|
|
|