[tor-commits] [bridgedb/master] Add methods to bridgedb.bridges.Bridge for constructing bridge lines.
isis at torproject.org
isis at torproject.org
Sat Mar 21 02:02:58 UTC 2015
commit 50a01e2d6fc22603f305f3c44acda67b259f5ced
Author: Isis Lovecruft <isis at torproject.org>
Date: Sat Dec 6 01:42:02 2014 +0000
Add methods to bridgedb.bridges.Bridge for constructing bridge lines.
* ADD new methods and properties:
_constructBridgeLine()
_getTransportForRequest()
_getVanillaForRequest()
allVanillaAddresses()
getBridgeLine()
supportedTransportTypes()
* CHANGE _updateORAddresses() to take a three-tuple.
* ADD new PluggableTransportUnavailable exception class for when a
pluggable transport has been requested which the Bridge doesn't know
about.
---
lib/bridgedb/bridges.py | 218 +++++++++++++++++++++++++++++++++++++++++++++--
1 file changed, 212 insertions(+), 6 deletions(-)
diff --git a/lib/bridgedb/bridges.py b/lib/bridgedb/bridges.py
index cb81dff..640008f 100644
--- a/lib/bridgedb/bridges.py
+++ b/lib/bridgedb/bridges.py
@@ -27,11 +27,14 @@ from bridgedb.parse.fingerprint import toHex
from bridgedb.parse.fingerprint import fromHex
+class PluggableTransportUnavailable(Exception):
+ """Raised when a :class:`Bridge` doesn't have the requested
+ :class:`PluggableTransport`.
+ """
class MalformedBridgeInfo(ValueError):
"""Raised when some information about a bridge appears malformed."""
-
class MalformedPluggableTransport(MalformedBridgeInfo):
"""Raised when information used to initialise a :class:`PluggableTransport`
appears malformed.
@@ -653,14 +656,170 @@ class Bridge(object):
"Actual descriptor digest: %s\n") %
(descriptor.fingerprint, self.descriptorDigest, digested))
+ def _constructBridgeLine(self, addrport, includeFingerprint=True,
+ bridgePrefix=False):
+ """Construct a :term:`Bridge Line` from an (address, port) tuple.
+
+ :param tuple addrport: A 3-tuple of ``(address, port, ipversion)``
+ where ``address`` is a string, ``port`` is an integer, and
+ ``ipversion`` is a integer (``4`` or ``6``).
+ :param bool includeFingerprint: If ``True``, include the
+ ``fingerprint`` of this :class:`Bridge` in the returned bridge
+ line.
+ :param bool bridgePrefix: if ``True``, prefix the :term:`Bridge Line`
+ with ``'Bridge '``.
+ :raises MalformedBridgeInfo: if the **addrport** didn't turn out to be
+ a 2-tuple containing ``(ipaddress, port)``.
+ :rtype: string
+ :returns: A bridge line suitable for adding into a ``torrc`` file or
+ Tor Launcher.
+ """
+ if not addrport:
+ return
+
+ try:
+ address, port, version = addrport
+ except (TypeError, ValueError):
+ raise MalformedBridgeInfo("Can't process addrport: %r" % addrport)
+
+ bridgeLine = []
+
+ if bridgePrefix:
+ bridgeLine.append('Bridge')
+
+ if version == 4:
+ bridgeLine.append("%s:%d" % (str(address), port))
+ elif version == 6:
+ bridgeLine.append("[%s]:%d" % (str(address), port))
+ else:
+ raise MalformedBridgeInfo("IP version must be 4 or 6")
+
+ if includeFingerprint:
+ bridgeLine.append("%s" % self.fingerprint)
+
+ return ' '.join(bridgeLine)
+
+ def _getTransportForRequest(self, bridgeRequest):
+ """If a transport was requested, return the correlated
+ :term:`Bridge Line` based upon the client identifier in the
+ **bridgeRequest**.
+
+ :type bridgeRequest: :class:`bridgedb.bridgerequest.BridgeRequestBase`
+ :param bridgeRequest: A ``BridgeRequest`` which stores all of the
+ client-specified options for which type of bridge they want to
+ receive.
+ :raises PluggableTransportUnavailable: if this bridge doesn't have any
+ of the requested pluggable transport type. This shouldn't happen
+ because the bridges are filtered into the client's hashring based
+ on the **bridgeRequest** options, however, this is useful in the
+ unlikely event that it does happen, so that the calling function
+ can fetch an additional bridge from the hashring as recompense for
+ what would've otherwise been a missing :term:`Bridge Line`.
+ :rtype: str or ``None``
+ :returns: If no transports were requested, return ``None``, otherwise
+ return a :term:`Bridge Line` for the requested pluggable transport
+ type.
+ """
+ addressClass = bridgeRequest.addressClass
+ desiredTransport = bridgeRequest.justOnePTType()
+ hashringPosition = bridgeRequest.getHashringPlacement(bridgeRequest.client,
+ 'Order-Or-Addresses')
+
+ logging.info("Bridge %s answering request for %s transport..." %
+ (safelog.logSafely(self.fingerprint), desiredTransport))
+ # Filter all this Bridge's ``transports`` according to whether or not
+ # their ``methodname`` matches the requested transport, i.e. only
+ # 'obfs3' transports, or only 'scramblesuit' transports:
+ transports = filter(lambda pt: desiredTransport == pt.methodname,
+ self.transports)
+ # Filter again for whichever of IPv4 or IPv6 was requested:
+ transports = filter(lambda pt: isinstance(pt.address, addressClass),
+ transports)
+
+ if transports:
+ return transports[hashringPosition % len(transports)]
+ else:
+ raise PluggableTransportUnavailable(
+ ("Client requested transport %s from bridge %s, but this "
+ "bridge doesn't have any of that transport!") %
+ (desiredTransport, self.fingerprint))
+
+ def _getVanillaForRequest(self, bridgeRequest):
+ """If vanilla bridges were requested, return the assigned
+ :term:`Bridge Line` based upon the client identifier in the
+ **bridgeRequest**.
+
+ :type bridgeRequest: :class:`bridgedb.bridgerequest.BridgeRequestBase`
+ :param bridgeRequest: A ``BridgeRequest`` which stores all of the
+ client-specified options for which type of bridge they want to
+ receive.
+ :rtype: str or ``None``
+ :returns: If no transports were requested, return ``None``, otherwise
+ return a :term:`Bridge Line` for the requested pluggable transport
+ type.
+ """
+ logging.info("Bridge %s answering request for vanilla address..." % self)
+
+ if not bridgeRequest.filters:
+ logging.debug(("Request %s didn't have any filters; "
+ "generating them now...") % bridgeRequest)
+ bridgeRequest.generateFilters()
+
+ addresses = self.allVanillaAddresses
+
+ # Filter ``allVanillaAddresses`` by whether IPv4 or IPv6 was requested:
+ addresses = filter(
+ # ``address`` here is a 3-tuple:
+ # ``(ipaddr.IPAddress, int(port), int(ipaddr.IPAddress.version))``
+ lambda address: isinstance(address[0], bridgeRequest.addressClass),
+ self.allVanillaAddresses)
+
+ if addresses:
+ # Use the client's unique data to HMAC them into their position in
+ # the hashring of filtered bridges addresses:
+ position = bridgeRequest.getHashringPlacement('Order-Or-Addresses',
+ bridgeRequest.client)
+ logging.debug("Client's hashring position is %r" % position)
+ vanilla = addresses[position % len(addresses)]
+ logging.info("Got vanilla bridge for client.")
+
+ return vanilla
+
def _updateORAddresses(self, orAddresses):
+ """Update this :class:`Bridge`'s :data:`orAddresses` attribute from a
+ 3-tuple (i.e. as Stem creates when parsing descriptors).
+
+ :param tuple orAddresses: A 3-tuple of: an IP address, a port number,
+ and a boolean (``False`` if IPv4, ``True`` if IPv6).
+ :raises FutureWarning: if any IPv4 addresses are found. As of
+ tor-0.2.5, only IPv6 addresses should be found in a descriptor's
+ `ORAddress` line.
+ """
for (address, port, ipVersion) in orAddresses:
+ version = 6
if not ipVersion: # `False` means IPv4; `True` means IPv6.
# See https://bugs.torproject.org/9380#comment:27
- warnings.warn(FutureWarning(
- ("Got IPv4 address in 'a'/'or-address' line! "
- "Desriptor format may have changed!")))
- self.orAddresses.append(tuple([address, port]))
+ warnings.warn(FutureWarning((
+ "Got IPv4 address in 'a'/'or-address' line! Descriptor "
+ "format may have changed!")))
+ version = 4
+
+ validatedAddress = isIPAddress(address, compressed=False)
+ if validatedAddress:
+ self.orAddresses.append( (validatedAddress, port, version,) )
+
+ @property
+ def allVanillaAddresses(self):
+ """Get all valid, non-PT address:port pairs for this bridge.
+
+ :rtype: list
+ :returns: All of this bridge's ORAddresses, as well as its ORPort IP
+ address and port.
+ """
+ addresses = self.orAddresses
+ # Add the default ORPort address:
+ addresses.append((self.address, self.orPort,))
+ return addresses
def assertOK(self):
"""Perform some additional validation on this bridge's info.
@@ -697,6 +856,41 @@ class Bridge(object):
if malformed:
raise MalformedBridgeInfo('\n'.join(malformed))
+ def getBridgeLine(self, bridgeRequest, includeFingerprint=True,
+ bridgePrefix=False):
+ """Return a valid :term:`Bridge Line` for a client to give to Tor
+ Launcher or paste directly into their ``torrc``.
+
+ This is a helper method to call either :meth:`_getTransportForRequest`
+ or :meth:`_getVanillaForRequest` depending on whether or not any
+ :class:`PluggableTransport`s were requested in the
+ :class:`bridgeRequest <bridgedb bridgerequest.BridgeRequestBase>`, and
+ then construct the :term:`Bridge Line` accordingly.
+
+ :type bridgeRequest: :class:`bridgedb.bridgerequest.BridgeRequestBase`
+ :param bridgeRequest: A ``BridgeRequest`` which stores all of the
+ client-specified options for which type of bridge they want to
+ receive.
+ :param bool includeFingerprint: If ``True``, include the
+ ``fingerprint`` of this :class:`Bridge` in the returned bridge
+ line.
+ :param bool bridgePrefix: if ``True``, prefix the :term:`Bridge Line`
+ with ``'Bridge '``.
+ """
+ if not bridgeRequest.isValid():
+ logging.info("Bridge request was not valid. Dropping request.")
+ return # XXX raise error perhaps?
+
+ if bridgeRequest.transports:
+ pt = self._getTransportForRequest(bridgeRequest)
+ bridgeLine = pt.getTransportLine(includeFingerprint, bridgePrefix)
+ else:
+ addrport = self._getVanillaForRequest(bridgeRequest)
+ bridgeLine = self._constructBridgeLine(addrport,
+ includeFingerprint,
+ bridgePrefix)
+ return bridgeLine
+
def getDescriptorLastPublished(self):
"""Get the timestamp for when this bridge's last known server
descriptor was published.
@@ -733,7 +927,19 @@ class Bridge(object):
"""
return getattr(self.descriptors['networkstatus'], 'published', None)
- def updateFromNetworkstatus(self, descriptor):
+ @property
+ def supportedTransportTypes(self):
+ """A deduplicated list of all the :data:`PluggableTranport.methodname`s
+ which this bridge supports.
+ """
+ supported = []
+
+ for transport in self.transports:
+ supported.append(transport.methodname)
+
+ return list(set(supported))
+
+ def updateFromNetworkStatus(self, descriptor):
"""Update this bridge's attributes from a parsed networkstatus
descriptor.
More information about the tor-commits
mailing list