From 8b864c3eae91f906e28de1865b76c47a04d67c55 Mon Sep 17 00:00:00 2001 From: Brian Warner Date: Tue, 20 Dec 2016 20:34:42 -0500 Subject: [PATCH 1/2] parse/transmit/record hint priorities Use --transit-helper=tcp:HOST:PORT:priority=1.3 to override the default 0.0 . Larger (positive) priority numbers will be attempted first. --- src/wormhole/test/test_transit.py | 53 +++++++++++++++++++++---------- src/wormhole/transit.py | 40 +++++++++++++++-------- 2 files changed, 64 insertions(+), 29 deletions(-) diff --git a/src/wormhole/test/test_transit.py b/src/wormhole/test/test_transit.py index 4d1e53f..dee0e63 100644 --- a/src/wormhole/test/test_transit.py +++ b/src/wormhole/test/test_transit.py @@ -138,29 +138,29 @@ class Hints(unittest.TestCase): def test_endpoint_from_hint_obj(self): c = transit.Common("") efho = c._endpoint_from_hint_obj - self.assertIsInstance(efho(transit.DirectTCPV1Hint("host", 1234)), + self.assertIsInstance(efho(transit.DirectTCPV1Hint("host", 1234, 0.0)), endpoints.HostnameEndpoint) self.assertEqual(efho("unknown:stuff:yowza:pivlor"), None) # c._tor_manager is currently None - self.assertEqual(efho(transit.TorTCPV1Hint("host", "port")), None) + self.assertEqual(efho(transit.TorTCPV1Hint("host", "port", 0)), None) c._tor_manager = mock.Mock() def tor_ep(hostname, port): if hostname == "non-public": return None return ("tor_ep", hostname, port) c._tor_manager.get_endpoint_for = mock.Mock(side_effect=tor_ep) - self.assertEqual(efho(transit.DirectTCPV1Hint("host", 1234)), + self.assertEqual(efho(transit.DirectTCPV1Hint("host", 1234, 0.0)), ("tor_ep", "host", 1234)) - self.assertEqual(efho(transit.TorTCPV1Hint("host2.onion", 1234)), + self.assertEqual(efho(transit.TorTCPV1Hint("host2.onion", 1234, 0.0)), ("tor_ep", "host2.onion", 1234)) - self.assertEqual(efho(transit.DirectTCPV1Hint("non-public", 1234)), + self.assertEqual(efho(transit.DirectTCPV1Hint("non-public", 1234, 0.0)), None) self.assertEqual(efho(UnknownHint("foo")), None) def test_comparable(self): - h1 = transit.DirectTCPV1Hint("hostname", "port1") - h1b = transit.DirectTCPV1Hint("hostname", "port1") - h2 = transit.DirectTCPV1Hint("hostname", "port2") + h1 = transit.DirectTCPV1Hint("hostname", "port1", 0.0) + h1b = transit.DirectTCPV1Hint("hostname", "port1", 0.0) + h2 = transit.DirectTCPV1Hint("hostname", "port2", 0.0) r1 = transit.RelayV1Hint(tuple(sorted([h1, h2]))) r2 = transit.RelayV1Hint(tuple(sorted([h2, h1]))) r3 = transit.RelayV1Hint(tuple(sorted([h1b, h2]))) @@ -173,9 +173,15 @@ class Hints(unittest.TestCase): p = c._parse_tcp_v1_hint self.assertEqual(p({"type": "unknown"}), None) h = p({"type": "direct-tcp-v1", "hostname": "foo", "port": 1234}) - self.assertEqual(h, transit.DirectTCPV1Hint("foo", 1234)) + self.assertEqual(h, transit.DirectTCPV1Hint("foo", 1234, 0.0)) + h = p({"type": "direct-tcp-v1", "hostname": "foo", "port": 1234, + "priority": 2.5}) + self.assertEqual(h, transit.DirectTCPV1Hint("foo", 1234, 2.5)) h = p({"type": "tor-tcp-v1", "hostname": "foo", "port": 1234}) - self.assertEqual(h, transit.TorTCPV1Hint("foo", 1234)) + self.assertEqual(h, transit.TorTCPV1Hint("foo", 1234, 0.0)) + h = p({"type": "tor-tcp-v1", "hostname": "foo", "port": 1234, + "priority": 2.5}) + self.assertEqual(h, transit.TorTCPV1Hint("foo", 1234, 2.5)) self.assertEqual(p({"type": "direct-tcp-v1"}), None) # missing hostname self.assertEqual(p({"type": "direct-tcp-v1", "hostname": 12}), @@ -192,7 +198,11 @@ class Hints(unittest.TestCase): value = transit.parse_hint_argv(hint, stderr=stderr) return value, stderr.getvalue() h,stderr = p("tcp:host:1234") - self.assertEqual(h, transit.DirectTCPV1Hint("host", 1234)) + self.assertEqual(h, transit.DirectTCPV1Hint("host", 1234, 0.0)) + self.assertEqual(stderr, "") + + h,stderr = p("tcp:host:1234:priority=2.6") + self.assertEqual(h, transit.DirectTCPV1Hint("host", 1234, 2.6)) self.assertEqual(stderr, "") h,stderr = p("$!@#^") @@ -204,16 +214,26 @@ class Hints(unittest.TestCase): self.assertEqual(stderr, "unknown hint type 'unknown' in 'unknown:stuff'\n") + h,stderr = p("tcp:just-a-hostname") + self.assertEqual(h, None) + self.assertEqual(stderr, + "unparseable TCP hint (need more colons) 'tcp:just-a-hostname'\n") + h,stderr = p("tcp:host:number") self.assertEqual(h, None) self.assertEqual(stderr, - "unparseable TCP hint 'tcp:host:number'\n") + "non-numeric port in TCP hint 'tcp:host:number'\n") + + h,stderr = p("tcp:host:1234:priority=bad") + self.assertEqual(h, None) + self.assertEqual(stderr, + "non-float priority= in TCP hint 'tcp:host:1234:priority=bad'\n") def test_describe_hint_obj(self): d = transit.describe_hint_obj - self.assertEqual(d(transit.DirectTCPV1Hint("host", 1234)), + self.assertEqual(d(transit.DirectTCPV1Hint("host", 1234, 0.0)), "tcp:host:1234") - self.assertEqual(d(transit.TorTCPV1Hint("host", 1234)), + self.assertEqual(d(transit.TorTCPV1Hint("host", 1234, 0.0)), "tor:host:1234") self.assertEqual(d(UnknownHint("stuff")), str(UnknownHint("stuff"))) @@ -225,8 +245,9 @@ class Basic(unittest.TestCase): hints = yield c.get_connection_hints() self.assertEqual(hints, [{"type": "relay-v1", "hints": [{"type": "direct-tcp-v1", - "hostname": "host", - "port": 1234}], + "hostname": "host", + "port": 1234, + "priority": 0.0}], }]) self.assertRaises(InternalError, transit.Common, 123) diff --git a/src/wormhole/transit.py b/src/wormhole/transit.py index 284fb1b..20b8d88 100644 --- a/src/wormhole/transit.py +++ b/src/wormhole/transit.py @@ -87,8 +87,8 @@ def build_sided_relay_handshake(key, side): # * expect to see the receiver/sender handshake bytes from the other side # * the sender writes "go\n", the receiver waits for "go\n" # * the rest of the connection contains transit data -DirectTCPV1Hint = namedtuple("DirectTCPV1Hint", ["hostname", "port"]) -TorTCPV1Hint = namedtuple("TorTCPV1Hint", ["hostname", "port"]) +DirectTCPV1Hint = namedtuple("DirectTCPV1Hint", ["hostname", "port", "priority"]) +TorTCPV1Hint = namedtuple("TorTCPV1Hint", ["hostname", "port", "priority"]) # RelayV1Hint contains a tuple of DirectTCPV1Hint and TorTCPV1Hint hints (we # use a tuple rather than a list so they'll be hashable into a set). For each # one, make the TCP connection, send the relay handshake, then complete the @@ -106,6 +106,7 @@ def describe_hint_obj(hint): def parse_hint_argv(hint, stderr=sys.stderr): assert isinstance(hint, type(u"")) # return tuple or None for an unparseable hint + priority = 0.0 mo = re.search(r'^([a-zA-Z0-9]+):(.*)$', hint) if not mo: print("unparseable hint '%s'" % (hint,), file=stderr) @@ -115,17 +116,27 @@ def parse_hint_argv(hint, stderr=sys.stderr): print("unknown hint type '%s' in '%s'" % (hint_type, hint), file=stderr) return None hint_value = mo.group(2) - mo = re.search(r'^(.*):(\d+)$', hint_value) - if not mo: - print("unparseable TCP hint '%s'" % (hint,), file=stderr) + pieces = hint_value.split(":") + if len(pieces) < 2: + print("unparseable TCP hint (need more colons) '%s'" % (hint,), + file=stderr) return None - hint_host = mo.group(1) - try: - hint_port = int(mo.group(2)) - except ValueError: + mo = re.search(r'^(\d+)$', pieces[1]) + if not mo: print("non-numeric port in TCP hint '%s'" % (hint,), file=stderr) return None - return DirectTCPV1Hint(hint_host, hint_port) + hint_host = pieces[0] + hint_port = int(pieces[1]) + for more in pieces[2:]: + if more.startswith("priority="): + more_pieces = more.split("=") + try: + priority = float(more_pieces[1]) + except ValueError: + print("non-float priority= in TCP hint '%s'" % (hint,), + file=stderr) + return None + return DirectTCPV1Hint(hint_host, hint_port, priority) TIMEOUT=15 @@ -604,7 +615,7 @@ class Common: # some test hosts, including the appveyor VMs, *only* have # 127.0.0.1, and the tests will hang badly if we remove it. addresses = non_loopback_addresses - direct_hints = [DirectTCPV1Hint(six.u(addr), portnum) + direct_hints = [DirectTCPV1Hint(six.u(addr), portnum, 0.0) for addr in addresses] ep = endpoints.serverFromString(reactor, "tcp:%d" % portnum) return direct_hints, ep @@ -620,6 +631,7 @@ class Common: direct_hints = yield self._get_direct_hints() for dh in direct_hints: hints.append({u"type": u"direct-tcp-v1", + u"priority": dh.priority, u"hostname": dh.hostname, u"port": dh.port, # integer }) @@ -627,6 +639,7 @@ class Common: rhint = {u"type": u"relay-v1", u"hints": []} for rh in relay.hints: rhint[u"hints"].append({u"type": u"direct-tcp-v1", + u"priority": rh.priority, u"hostname": rh.hostname, u"port": rh.port}) hints.append(rhint) @@ -686,10 +699,11 @@ class Common: if not(u"port" in hint and isinstance(hint[u"port"], int)): log.msg("invalid port in hint: %r" % (hint,)) return None + priority = hint.get(u"priority", 0.0) if hint_type == u"direct-tcp-v1": - return DirectTCPV1Hint(hint[u"hostname"], hint[u"port"]) + return DirectTCPV1Hint(hint[u"hostname"], hint[u"port"], priority) else: - return TorTCPV1Hint(hint[u"hostname"], hint[u"port"]) + return TorTCPV1Hint(hint[u"hostname"], hint[u"port"], priority) def add_connection_hints(self, hints): for h in hints: # hint structs From b2fd899ac98f74c3596cf41b288089d79a6778f4 Mon Sep 17 00:00:00 2001 From: Brian Warner Date: Fri, 30 Dec 2016 23:23:06 -0500 Subject: [PATCH 2/2] prioritize relay connections closes #103 --- src/wormhole/test/test_transit.py | 50 +++++++++++++++++++++++++++++++ src/wormhole/transit.py | 10 +++++++ 2 files changed, 60 insertions(+) diff --git a/src/wormhole/test/test_transit.py b/src/wormhole/test/test_transit.py index dee0e63..62848b4 100644 --- a/src/wormhole/test/test_transit.py +++ b/src/wormhole/test/test_transit.py @@ -1374,6 +1374,56 @@ class Transit(unittest.TestCase): self._waiters[0].callback("winner") self.assertEqual(results, ["winner"]) + @inlineCallbacks + def test_priorities(self): + clock = task.Clock() + s = transit.TransitSender("", reactor=clock, no_listen=True) + s.set_transit_key(b"key") + hints = yield s.get_connection_hints() + del hints + s.add_connection_hints([ + {"type": "relay-v1", + "hints": [{"type": "direct-tcp-v1", + "hostname": "relay", "port": 1234}]}, + {"type": "direct-tcp-v1", + "hostname": "direct", "port": 1234}, + {"type": "relay-v1", + "hints": [{"type": "direct-tcp-v1", "priority": 2.0, + "hostname": "relay2", "port": 1234}, + {"type": "direct-tcp-v1", "priority": 3.0, + "hostname": "relay3", "port": 1234}]}, + {"type": "relay-v1", + "hints": [{"type": "direct-tcp-v1", "priority": 2.0, + "hostname": "relay4", "port": 1234}]}, + ]) + s._endpoint_from_hint_obj = self._endpoint_from_hint_obj + s._start_connector = self._start_connector + + d = s.connect() + results = [] + d.addBoth(results.append) + self.assertEqual(results, []) + # direct connector should be used first, then the priority=3.0 relay, + # then the two 2.0 relays, then the (default) 0.0 relay + + self.assertEqual(self._connectors, ["direct"]) + + clock.advance(s.RELAY_DELAY + 1.0) + self.assertEqual(self._connectors, ["direct", "relay3"]) + + clock.advance(s.RELAY_DELAY) + self.assertIn(self._connectors, + (["direct", "relay3", "relay2", "relay4"], + ["direct", "relay3", "relay4", "relay2"])) + + clock.advance(s.RELAY_DELAY) + self.assertIn(self._connectors, + (["direct", "relay3", "relay2", "relay4", "relay"], + ["direct", "relay3", "relay4", "relay2", "relay"])) + + self._waiters[0].callback("winner") + self.assertEqual(results, ["winner"]) + @inlineCallbacks def test_no_direct_hints(self): clock = task.Clock() diff --git a/src/wormhole/transit.py b/src/wormhole/transit.py index 20b8d88..4aa5972 100644 --- a/src/wormhole/transit.py +++ b/src/wormhole/transit.py @@ -821,8 +821,17 @@ class Common: # resolve quickly. Many direct hints will be to unused local-network # IP addresses, which won't answer, and would take the full TCP # timeout (30s or more) to fail. + + prioritized_relays = {} for rh in self._our_relay_hints: for hint_obj in rh.hints: + priority = hint_obj.priority + if priority not in prioritized_relays: + prioritized_relays[priority] = set() + prioritized_relays[priority].add(hint_obj) + + for priority in sorted(prioritized_relays, reverse=True): + for hint_obj in prioritized_relays[priority]: ep = self._endpoint_from_hint_obj(hint_obj) if not ep: continue @@ -831,6 +840,7 @@ class Common: self._start_connector, ep, description, is_relay=True) contenders.append(d) + relay_delay += self.RELAY_DELAY if not contenders: raise TransitError("No contenders for connection")