Rev 5249: Implement a thread that can re-raise exceptions. in file:///home/vila/src/bzr/experimental/leaking-tests/

Vincent Ladeuil v.ladeuil+lp at free.fr
Wed May 26 11:44:08 BST 2010


At file:///home/vila/src/bzr/experimental/leaking-tests/

------------------------------------------------------------
revno: 5249
revision-id: v.ladeuil+lp at free.fr-20100526104407-0w315whpmsdeafjk
parent: v.ladeuil+lp at free.fr-20100526103818-1x38taslbkp097z9
committer: Vincent Ladeuil <v.ladeuil+lp at free.fr>
branch nick: propagate-exceptions
timestamp: Wed 2010-05-26 12:44:07 +0200
message:
  Implement a thread that can re-raise exceptions.
  
  * bzrlib/tests/test_transport.py:
  (TestThreadWithException): Test that the execption is re-raised at
  join() time.
  
  * bzrlib/tests/test_server.py:
  (ThreadWithException): We need a thread that catch and re-raise
  any exception.
-------------- next part --------------
=== modified file 'bzrlib/tests/test_server.py'
--- a/bzrlib/tests/test_server.py	2010-05-24 01:05:14 +0000
+++ b/bzrlib/tests/test_server.py	2010-05-26 10:44:07 +0000
@@ -16,6 +16,8 @@
 
 import socket
 import select
+import sys
+import threading
 
 
 from bzrlib import (
@@ -227,6 +229,35 @@
         raise NotImplementedError
 
 
+class ThreadWithException(threading.Thread):
+    """A catching exception thread.
+
+    If an exception occurs during the thread execution, it's caught and
+    re-raised when the thread is joined().
+    """
+
+    def __init__(self, *args, **kwargs):
+        super(ThreadWithException, self).__init__(*args, **kwargs)
+        self.exception = None
+
+    def run(self):
+        """Overrides Thread.run to capture any exception."""
+        try:
+            super(ThreadWithException, self).run()
+        except Exception, e:
+            self.exception = sys.exc_info()
+
+    def join(self, *args, **kwargs):
+        """Overrides Thread.join to raise any exception caught."""
+        # Note that we don't care about the timeout parameter here: either the
+        # thread has raised an exception and it should be raised (and join()
+        # should succeed whatever the timeout is) or it's still alive which
+        # means it didn't encounter an exception.
+        super(ThreadWithException, self).join(*args, **kwargs)
+        if self.exception is not None:
+            raise self.exception
+
+
 class SmartTCPServer_for_testing(server.SmartTCPServer):
     """Server suitable for use by transport tests.
 

=== modified file 'bzrlib/tests/test_transport.py'
--- a/bzrlib/tests/test_transport.py	2010-02-23 07:43:11 +0000
+++ b/bzrlib/tests/test_transport.py	2010-05-26 10:44:07 +0000
@@ -989,3 +989,25 @@
         # And the rest are threads
         for t in started[1:]:
             t.join()
+
+
+class TestThreadWithException(tests.TestCase):
+
+    def test_start_and_join_smoke_test(self):
+        def do_nothing():
+            pass
+
+        tt = test_server.ThreadWithException(target=do_nothing)
+        tt.start()
+        tt.join()
+
+    def test_exception_is_re_raised(self):
+        class MyException(Exception):
+            pass
+
+        def raise_my_exception():
+            raise MyException()
+
+        tt = test_server.ThreadWithException(target=raise_my_exception)
+        tt.start()
+        self.assertRaises(MyException, tt.join)



More information about the bazaar-commits mailing list