summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorDamian Johnson <atagar@torproject.org>2018-08-15 10:55:25 -0700
committerDamian Johnson <atagar@torproject.org>2018-08-15 11:02:23 -0700
commitabc6f298e456f73c8562bf23f2c4ad48dbd0f18b (patch)
tree97d16306a2c6a3142723e727ad999cb8aa32105b
parentcb5a49602ffb6fb04c2c32d0866cd0a331b5e9ab (diff)
Close time window in event handling unit test
As Dave discussed on #27053 these new unit tests occasionally fail. Think I see what's up: our emit_event() helper blocked until events were consumed but *not* until they were handled. That is to say in the following psudocode... event_loop function: while True: event = event_queue.pop() # <=== right here for listener in listeners: listener(event) ... our tests could make assertions after dequeueing events but before calling the listener, producing the assertion failures he saw. Joining on the event thread to ensure it's done performing any work it has. On the upside this actually simplifies our test code too! :P Ran the tests several dozen times and this seems to do the trick for me but unforuntately since this these test failures could be pretty rare tough to be sure.
-rw-r--r--test/unit/control/controller.py107
1 files changed, 44 insertions, 63 deletions
diff --git a/test/unit/control/controller.py b/test/unit/control/controller.py
index 156e62c5..7ee2ab70 100644
--- a/test/unit/control/controller.py
+++ b/test/unit/control/controller.py
@@ -625,30 +625,21 @@ class TestControl(unittest.TestCase):
self.assertRaises(InvalidRequest, self.controller.add_event_listener, Mock(), EventType.SIGNAL)
@patch('time.time', Mock(return_value = TEST_TIMESTAMP))
- @patch('stem.control.Controller.is_alive')
- def test_events_get_received(self, is_alive_mock):
+ def test_events_get_received(self):
"""
Trigger an event, checking that our listeners get notified.
"""
- is_alive_mock.return_value = True
- self.controller._launch_threads()
+ self._emit_event(CIRC_EVENT)
+ self.circ_listener.assert_called_once_with(CIRC_EVENT)
+ self.bw_listener.assert_not_called()
+ self.malformed_listener.assert_not_called()
- try:
- self._emit_event(CIRC_EVENT)
- self.circ_listener.assert_called_once_with(CIRC_EVENT)
- self.bw_listener.assert_not_called()
- self.malformed_listener.assert_not_called()
-
- self._emit_event(BW_EVENT)
- self.bw_listener.assert_called_once_with(BW_EVENT)
- finally:
- is_alive_mock.return_value = False
- self.controller._close()
+ self._emit_event(BW_EVENT)
+ self.bw_listener.assert_called_once_with(BW_EVENT)
@patch('time.time', Mock(return_value = TEST_TIMESTAMP))
- @patch('stem.control.Controller.is_alive')
- def test_event_listing_with_error(self, is_alive_mock):
+ def test_event_listing_with_error(self):
"""
Raise an exception in an event listener to confirm it doesn't break our
event thread.
@@ -656,32 +647,21 @@ class TestControl(unittest.TestCase):
self.circ_listener.side_effect = ValueError('boom')
- is_alive_mock.return_value = True
- self.controller._launch_threads()
+ self._emit_event(CIRC_EVENT)
+ self.circ_listener.assert_called_once_with(CIRC_EVENT)
+ self.bw_listener.assert_not_called()
+ self.malformed_listener.assert_not_called()
- try:
- self._emit_event(CIRC_EVENT)
- self.circ_listener.assert_called_once_with(CIRC_EVENT)
- self.bw_listener.assert_not_called()
- self.malformed_listener.assert_not_called()
-
- self._emit_event(BW_EVENT)
- self.bw_listener.assert_called_once_with(BW_EVENT)
- finally:
- is_alive_mock.return_value = False
- self.controller._close()
+ self._emit_event(BW_EVENT)
+ self.bw_listener.assert_called_once_with(BW_EVENT)
@patch('time.time', Mock(return_value = TEST_TIMESTAMP))
- @patch('stem.control.Controller.is_alive')
- def test_event_listing_with_malformed_event(self, is_alive_mock):
+ def test_event_listing_with_malformed_event(self):
"""
Attempt to parse a malformed event emitted from Tor. It's important this
doesn't break our event thread.
"""
- is_alive_mock.return_value = True
- self.controller._launch_threads()
-
# When stem.response.convert() encounters malformed content we still recast
# the message.
@@ -689,17 +669,13 @@ class TestControl(unittest.TestCase):
setattr(expected_bad_event, 'arrived_at', TEST_TIMESTAMP)
expected_bad_event.__class__ = stem.response.events.BandwidthEvent
- try:
- self._emit_event(BAD_EVENT)
- self.circ_listener.assert_not_called()
- self.bw_listener.assert_not_called()
- self.malformed_listener.assert_called_once_with(expected_bad_event)
+ self._emit_event(BAD_EVENT)
+ self.circ_listener.assert_not_called()
+ self.bw_listener.assert_not_called()
+ self.malformed_listener.assert_called_once_with(expected_bad_event)
- self._emit_event(BW_EVENT)
- self.bw_listener.assert_called_once_with(BW_EVENT)
- finally:
- is_alive_mock.return_value = False
- self.controller._close()
+ self._emit_event(BW_EVENT)
+ self.bw_listener.assert_called_once_with(BW_EVENT)
@patch('stem.control.Controller.get_version', Mock(return_value = stem.version.Version('0.5.0.14')))
@patch('stem.control.Controller.msg', Mock(return_value = ControlMessage.from_str('250 OK\r\n')))
@@ -827,21 +803,26 @@ class TestControl(unittest.TestCase):
self.controller.drop_guards()
def _emit_event(self, event):
- # Converting an event back into an uncast ControlMessage, then feeding it
- # into our controller's event queue.
-
- uncast_event = ControlMessage.from_str(event.raw_content())
- self.controller._event_queue.put(uncast_event)
- self.controller._event_notice.set()
-
- # Wait for the event to get asynchronously consumed. This should happen
- # right away, but if the system is bogged down it may take quite a few
- # milliseconds.
-
- event_sent_at = time.time()
-
- while not self.controller._event_queue.empty():
- time.sleep(0.01)
-
- if time.time() - event_sent_at > 5:
- self.fail('Event did not get processed after five seconds')
+ # Spins up our Controller's thread pool, emits an event, then shuts it
+ # down. This last part is important for a couple reasons...
+ #
+ # 1. So we don't leave any lingering threads.
+ #
+ # 2. To ensure our event handlers are done being executed. Events are
+ # processed asynchronously, so the only way to endsure it's done
+ # with its work is to join on the thread.
+
+ with patch('stem.control.Controller.is_alive') as is_alive_mock:
+ is_alive_mock.return_value = True
+ self.controller._launch_threads()
+
+ try:
+ # Converting an event back into an uncast ControlMessage, then feeding it
+ # into our controller's event queue.
+
+ uncast_event = ControlMessage.from_str(event.raw_content())
+ self.controller._event_queue.put(uncast_event)
+ self.controller._event_notice.set()
+ finally:
+ is_alive_mock.return_value = False
+ self.controller._close()