[tor-commits] [stem/master] Throwing DescriptorUnavailable when tor can't provide one for a relay
atagar at torproject.org
atagar at torproject.org
Fri Dec 5 01:29:27 UTC 2014
commit 6a3882a372149ff18b5be7f82fa4b9e5cac113b1
Author: Damian Johnson <atagar at torproject.org>
Date: Thu Dec 4 17:30:59 2014 -0800
Throwing DescriptorUnavailable when tor can't provide one for a relay
Tor gives a pretty sucky 'GETINFO request contained unrecognized keywords:
ns/id/5AC9C5AA75BA1F18D8459B326B4B8111A856D290' error when the descriptor is
unavailable for a given relay. Having us give something better...
https://trac.torproject.org/projects/tor/ticket/13879
---
docs/change_log.rst | 1 +
stem/__init__.py | 8 ++++++++
stem/control.py | 40 ++++++++++++++++++++++++++++++++--------
3 files changed, 41 insertions(+), 8 deletions(-)
diff --git a/docs/change_log.rst b/docs/change_log.rst
index 4034884..e9fcac7 100644
--- a/docs/change_log.rst
+++ b/docs/change_log.rst
@@ -52,6 +52,7 @@ The following are only available within Stem's `git repository
* :func:`~stem.process.launch_tor_with_config` could cause a "Too many open files" OSError if called too many times (:trac:`13141`)
* The :func:`~stem.control.Controller.get_exit_policy` method errored if tor couldn't determine our external address
* The Controller's methods for retrieving descriptors could raise unexpected ValueErrors if tor didn't have any descriptors available
+ * Throwing a new :class:`~stem.DescriptorUnavailable` exception type when the :class:`~stem.control.Controller` can't provide the descriptor for a relay (:trac:`13879`)
* **Descriptors**
diff --git a/stem/__init__.py b/stem/__init__.py
index 310f4ab..ffdfe72 100644
--- a/stem/__init__.py
+++ b/stem/__init__.py
@@ -13,6 +13,7 @@ Library for working with the tor process.
|- OperationFailed - Tor was unable to successfully complete the operation.
| |- UnsatisfiableRequest - Tor was unable to satisfy a valid request.
| | +- CircuitExtensionFailed - Attempt to make or extend a circuit failed.
+ | |- DescriptorUnavailable - The given relay descriptor is unavailable.
| +- InvalidRequest - Invalid request.
| +- InvalidArguments - Invalid request parameters.
+- SocketError - Communication with the socket failed.
@@ -459,6 +460,7 @@ __all__ = [
'OperationFailed',
'UnsatisfiableRequest',
'CircuitExtensionFailed',
+ 'DescriptorUnavailable',
'InvalidRequest',
'InvalidArguments',
'SocketError',
@@ -539,6 +541,12 @@ class CircuitExtensionFailed(UnsatisfiableRequest):
self.circ = circ
+class DescriptorUnavailable(OperationFailed):
+ """
+ Tor was unable to provide a descriptor for the given relay.
+ """
+
+
class InvalidRequest(OperationFailed):
"""
Exception raised when the request was invalid or malformed.
diff --git a/stem/control.py b/stem/control.py
index f947aca..cace3ad 100644
--- a/stem/control.py
+++ b/stem/control.py
@@ -1456,6 +1456,8 @@ class Controller(BaseController):
:returns: :class:`~stem.descriptor.microdescriptor.Microdescriptor` for the given relay
:raises:
+ * :class:`stem.DescriptorUnavailable` if unable to provide a descriptor
+ for the given relay
* :class:`stem.ControllerError` if unable to query the descriptor
* **ValueError** if **relay** doesn't conform with the pattern for being
a fingerprint or nickname
@@ -1476,10 +1478,16 @@ class Controller(BaseController):
else:
raise ValueError("'%s' isn't a valid fingerprint or nickname" % relay)
- desc_content = self.get_info(query, get_bytes = True)
+ try:
+ desc_content = self.get_info(query, get_bytes = True)
+ except stem.InvalidArguments as exc:
+ if str(exc).startswith('GETINFO request contained unrecognized keywords:'):
+ raise stem.DescriptorUnavailable("Tor was unable to provide the descriptor for '%s'" % relay)
+ else:
+ raise exc
if not desc_content:
- raise stem.ControllerError("Descriptor information is unavailable, tor might still be downloading it")
+ raise stem.DescriptorUnavailable("Descriptor information is unavailable, tor might still be downloading it")
return stem.descriptor.microdescriptor.Microdescriptor(desc_content)
@@ -1555,6 +1563,8 @@ class Controller(BaseController):
:returns: :class:`~stem.descriptor.server_descriptor.RelayDescriptor` for the given relay
:raises:
+ * :class:`stem.DescriptorUnavailable` if unable to provide a descriptor
+ for the given relay
* :class:`stem.ControllerError` if unable to query the descriptor
* **ValueError** if **relay** doesn't conform with the pattern for being
a fingerprint or nickname
@@ -1576,10 +1586,16 @@ class Controller(BaseController):
else:
raise ValueError("'%s' isn't a valid fingerprint or nickname" % relay)
- desc_content = self.get_info(query, get_bytes = True)
+ try:
+ desc_content = self.get_info(query, get_bytes = True)
+ except stem.InvalidArguments as exc:
+ if str(exc).startswith('GETINFO request contained unrecognized keywords:'):
+ raise stem.DescriptorUnavailable("Tor was unable to provide the descriptor for '%s'" % relay)
+ else:
+ raise exc
if not desc_content:
- raise stem.ControllerError("Descriptor information is unavailable, tor might still be downloading it")
+ raise stem.DescriptorUnavailable("Descriptor information is unavailable, tor might still be downloading it")
return stem.descriptor.server_descriptor.RelayDescriptor(desc_content)
except Exception as exc:
@@ -1622,7 +1638,7 @@ class Controller(BaseController):
if not self._is_server_descriptors_available():
raise stem.ControllerError(SERVER_DESCRIPTORS_UNSUPPORTED)
else:
- raise stem.ControllerError("Descriptor information is unavailable, tor might still be downloading it")
+ raise stem.DescriptorUnavailable("Descriptor information is unavailable, tor might still be downloading it")
for desc in stem.descriptor.server_descriptor._parse_file(io.BytesIO(desc_content)):
yield desc
@@ -1670,6 +1686,8 @@ class Controller(BaseController):
for the given relay
:raises:
+ * :class:`stem.DescriptorUnavailable` if unable to provide a descriptor
+ for the given relay
* :class:`stem.ControllerError` if unable to query the descriptor
* **ValueError** if **relay** doesn't conform with the pattern for being
a fingerprint or nickname
@@ -1695,10 +1713,16 @@ class Controller(BaseController):
else:
raise ValueError("'%s' isn't a valid fingerprint or nickname" % relay)
- desc_content = self.get_info(query, get_bytes = True)
+ try:
+ desc_content = self.get_info(query, get_bytes = True)
+ except stem.InvalidArguments as exc:
+ if str(exc).startswith('GETINFO request contained unrecognized keywords:'):
+ raise stem.DescriptorUnavailable("Tor was unable to provide the descriptor for '%s'" % relay)
+ else:
+ raise exc
if not desc_content:
- raise stem.ControllerError("Descriptor information is unavailable, tor might still be downloading it")
+ raise stem.DescriptorUnavailable("Descriptor information is unavailable, tor might still be downloading it")
if self.get_conf('UseMicrodescriptors', '0') == '1':
return stem.descriptor.router_status_entry.RouterStatusEntryMicroV3(desc_content)
@@ -1747,7 +1771,7 @@ class Controller(BaseController):
desc_content = self.get_info('ns/all', get_bytes = True)
if not desc_content:
- raise stem.ControllerError("Descriptor information is unavailable, tor might still be downloading it")
+ raise stem.DescriptorUnavailable("Descriptor information is unavailable, tor might still be downloading it")
desc_iterator = stem.descriptor.router_status_entry._parse_file(
io.BytesIO(desc_content),
More information about the tor-commits
mailing list