changeset: 89925:df6a6951b2c9 branch: 3.4 parent: 89923:619331c67638 user: Richard Oudkerk date: Sun Mar 23 12:30:54 2014 +0000 files: Lib/multiprocessing/managers.py Lib/multiprocessing/pool.py Lib/test/_test_multiprocessing.py Misc/NEWS description: Issue #20980: Stop wrapping exception when using ThreadPool. diff -r 619331c67638 -r df6a6951b2c9 Lib/multiprocessing/managers.py --- a/Lib/multiprocessing/managers.py Sun Mar 23 11:54:15 2014 +0000 +++ b/Lib/multiprocessing/managers.py Sun Mar 23 12:30:54 2014 +0000 @@ -1077,17 +1077,22 @@ )) -PoolProxy = MakeProxyType('PoolProxy', ( +BasePoolProxy = MakeProxyType('PoolProxy', ( 'apply', 'apply_async', 'close', 'imap', 'imap_unordered', 'join', - 'map', 'map_async', 'starmap', 'starmap_async', 'terminate' + 'map', 'map_async', 'starmap', 'starmap_async', 'terminate', )) -PoolProxy._method_to_typeid_ = { +BasePoolProxy._method_to_typeid_ = { 'apply_async': 'AsyncResult', 'map_async': 'AsyncResult', 'starmap_async': 'AsyncResult', 'imap': 'Iterator', 'imap_unordered': 'Iterator' } +class PoolProxy(BasePoolProxy): + def __enter__(self): + return self + def __exit__(self, exc_type, exc_val, exc_tb): + self.terminate() # # Definition of SyncManager diff -r 619331c67638 -r df6a6951b2c9 Lib/multiprocessing/pool.py --- a/Lib/multiprocessing/pool.py Sun Mar 23 11:54:15 2014 +0000 +++ b/Lib/multiprocessing/pool.py Sun Mar 23 12:30:54 2014 +0000 @@ -90,7 +90,8 @@ return "" % str(self) -def worker(inqueue, outqueue, initializer=None, initargs=(), maxtasks=None): +def worker(inqueue, outqueue, initializer=None, initargs=(), maxtasks=None, + wrap_exception=False): assert maxtasks is None or (type(maxtasks) == int and maxtasks > 0) put = outqueue.put get = inqueue.get @@ -117,7 +118,8 @@ try: result = (True, func(*args, **kwds)) except Exception as e: - e = ExceptionWithTraceback(e, e.__traceback__) + if wrap_exception: + e = ExceptionWithTraceback(e, e.__traceback__) result = (False, e) try: put((job, i, result)) @@ -137,6 +139,8 @@ ''' Class which supports an async version of applying functions to arguments. ''' + _wrap_exception = True + def Process(self, *args, **kwds): return self._ctx.Process(*args, **kwds) @@ -220,7 +224,8 @@ w = self.Process(target=worker, args=(self._inqueue, self._outqueue, self._initializer, - self._initargs, self._maxtasksperchild) + self._initargs, self._maxtasksperchild, + self._wrap_exception) ) self._pool.append(w) w.name = w.name.replace('Process', 'PoolWorker') @@ -736,6 +741,7 @@ # class ThreadPool(Pool): + _wrap_exception = False @staticmethod def Process(*args, **kwds): diff -r 619331c67638 -r df6a6951b2c9 Lib/test/_test_multiprocessing.py --- a/Lib/test/_test_multiprocessing.py Sun Mar 23 11:54:15 2014 +0000 +++ b/Lib/test/_test_multiprocessing.py Sun Mar 23 12:30:54 2014 +0000 @@ -1810,6 +1810,17 @@ self.assertIn('raise RuntimeError(123) # some comment', f1.getvalue()) + @classmethod + def _test_wrapped_exception(cls): + raise RuntimeError('foo') + + def test_wrapped_exception(self): + # Issue #20980: Should not wrap exception when using thread pool + with self.Pool(1) as p: + with self.assertRaises(RuntimeError): + p.apply(self._test_wrapped_exception) + + def raising(): raise KeyError("key") diff -r 619331c67638 -r df6a6951b2c9 Misc/NEWS --- a/Misc/NEWS Sun Mar 23 11:54:15 2014 +0000 +++ b/Misc/NEWS Sun Mar 23 12:30:54 2014 +0000 @@ -21,6 +21,8 @@ Library ------- +- Issue #20980: Stop wrapping exception when using ThreadPool. + - Issue #20990: Fix issues found by pyflakes for multiprocessing. - Issue #21015: SSL contexts will now automatically select an elliptic