[tor-commits] [stem/master] Letting add_event_listener() work when unauthenticated
atagar at torproject.org
atagar at torproject.org
Wed Jan 16 17:30:07 UTC 2013
commit 5d9c25300e6783b092a64fc66877f39566fc4a4f
Author: Damian Johnson <atagar at torproject.org>
Date: Wed Jan 16 09:21:18 2013 -0800
Letting add_event_listener() work when unauthenticated
If we called add_event_listener() prior to authentication then we'd throw an
error when calling get_version() to figure out if we meet the requirements.
Changing the behavior so that we check requirements and issue SETEVENTS if
able, but if not then simply enquing the listener. It'll then be attached
during our post-authentication hook.
The change that I'm more interested in this though is that our
post-authentication hook is less likely to go belly up. If we, say, added a
listener then attached the Controller to an older tor instance then our
SETEVENTS call could fail. This was an all-or-nothing call. :(
Changed it so that the hook will re-attach the events that it can, then warn
about the rest.
---
stem/control.py | 79 ++++++++++++++++++++++++++++++++-------
test/unit/control/controller.py | 5 ++
2 files changed, 70 insertions(+), 14 deletions(-)
diff --git a/stem/control.py b/stem/control.py
index 35d9701..8d6f1b7 100644
--- a/stem/control.py
+++ b/stem/control.py
@@ -139,6 +139,7 @@ import stem.descriptor.router_status_entry
import stem.descriptor.server_descriptor
import stem.exit_policy
import stem.response
+import stem.response.events
import stem.socket
import stem.util.connection
import stem.util.enum
@@ -1420,16 +1421,24 @@ class Controller(BaseController):
"""
# first checking that tor supports these event types
- for event_type in events:
- event_version = stem.response.events.EVENT_TYPE_TO_CLASS[event_type]._VERSION_ADDED
- if not self.get_version().meets_requirements(event_version):
- raise stem.InvalidRequest(552, "%s event requires Tor version %s or later" % (event_type, event_version))
-
with self._event_listeners_lock:
+ if self.is_authenticated():
+ for event_type in events:
+ event_version = stem.response.events.EVENT_TYPE_TO_CLASS[event_type]._VERSION_ADDED
+
+ if not self.get_version().meets_requirements(event_version):
+ raise stem.InvalidRequest(552, "%s event requires Tor version %s or later" % (event_type, event_version))
+
for event_type in events:
self._event_listeners.setdefault(event_type, []).append(listener)
- self._attach_listeners()
+ failed_events = self._attach_listeners()[1]
+
+ # restricted the failures to just things we requested
+ failed_events = set(failed_events).intersection(set(events))
+
+ if failed_events:
+ raise stem.ProtocolError("SETEVENTS rejected %s" % ", ".join(failed_events))
def remove_event_listener(self, listener):
"""
@@ -1949,10 +1958,19 @@ class Controller(BaseController):
# try to re-attach event listeners to the new instance
- try:
- self._attach_listeners()
- except stem.ProtocolError, exc:
- log.warn("We were unable to re-attach our event listeners to the new tor instance (%s)" % exc)
+ with self._event_listeners_lock:
+ try:
+ failed_events = self._attach_listeners()[1]
+
+ if failed_events:
+ # remove our listeners for these so we don't keep failing
+ for event_type in failed_events:
+ del self._event_listeners[event_type]
+
+ logging_id = "stem.controller.event_reattach-%s" % "-".join(failed_events)
+ log.log_once(logging_id, log.WARN, "We were unable to re-attach our event listeners to the new tor instance for: %s" % ", ".join(failed_events))
+ except stem.ProtocolError, exc:
+ log.warn("Unable to issue the SETEVENTS request to re-attach our listeners (%s)" % exc)
# issue TAKEOWNERSHIP if we're the owning process for this tor instance
@@ -1983,14 +2001,47 @@ class Controller(BaseController):
listener(event_message)
def _attach_listeners(self):
- # issues the SETEVENTS call for our event listeners
+ """
+ Attempts to subscribe to the self._event_listeners events from tor. This is
+ a no-op if we're not presently authenticated.
+
+ :returns: tuple of the form (set_events, failed_events)
+
+ :raises: :class:`stem.ControllerError` if unable to make our request to tor
+ """
+
+ set_events, failed_events = [], []
with self._event_listeners_lock:
- if self.is_alive():
+ if self.is_authenticated():
+ # try to set them all
response = self.msg("SETEVENTS %s" % " ".join(self._event_listeners.keys()))
- if not response.is_ok():
- raise stem.ProtocolError("SETEVENTS received unexpected response\n%s" % response)
+ if response.is_ok():
+ set_events = self._event_listeners.keys()
+ else:
+ # One of the following likely happened...
+ #
+ # * Our user attached listeners before having an authenticated
+ # connection, so we couldn't check if we met the version
+ # requirement.
+ #
+ # * User attached listeners to one tor instance, then connected us to
+ # an older tor instancce.
+ #
+ # * Some other controller hiccup (far less likely).
+ #
+ # See if we can set some subset of our events.
+
+ for event in self._event_listeners.keys():
+ response = self.msg("SETEVENTS %s" % " ".join(set_events + [event]))
+
+ if response.is_ok():
+ set_events.append(event)
+ else:
+ failed_events.append(event)
+
+ return (set_events, failed_events)
def _parse_circ_path(path):
diff --git a/test/unit/control/controller.py b/test/unit/control/controller.py
index 77c8f84..bf910f2 100644
--- a/test/unit/control/controller.py
+++ b/test/unit/control/controller.py
@@ -18,7 +18,10 @@ from test import mocking
class TestControl(unittest.TestCase):
def setUp(self):
socket = stem.socket.ControlSocket()
+
+ mocking.mock_method(Controller, "add_event_listener", mocking.no_op())
self.controller = Controller(socket, enable_caching = True)
+ mocking.revert_mocking()
def tearDown(self):
mocking.revert_mocking()
@@ -276,6 +279,8 @@ class TestControl(unittest.TestCase):
"""
# set up for failure to create any events
+ mocking.mock_method(Controller, "is_authenticated", mocking.return_true())
+ mocking.mock_method(Controller, "_attach_listeners", mocking.return_value(([], [])))
mocking.mock_method(Controller, "get_version", mocking.return_value(stem.version.Version('0.1.0.14')))
self.assertRaises(InvalidRequest, self.controller.add_event_listener, mocking.no_op(), EventType.BW)
More information about the tor-commits
mailing list