summaryrefslogtreecommitdiffstats
path: root/Lib/unittest
diff options
context:
space:
mode:
authorAntoine Pitrou <solipsis@pitrou.net>2015-03-18 23:01:37 (GMT)
committerAntoine Pitrou <solipsis@pitrou.net>2015-03-18 23:01:37 (GMT)
commit8eef6a9ad04f6f81190f44ae3ded427e4083baa2 (patch)
tree1efe0d3c609b40de1725719968942f6780f7581f /Lib/unittest
parent52a05ab52467ad1df7572f656cf87fc47aa0282d (diff)
parentd833779ceaebeb29352488ffddabf5fc2f070364 (diff)
downloadcpython-8eef6a9ad04f6f81190f44ae3ded427e4083baa2.zip
cpython-8eef6a9ad04f6f81190f44ae3ded427e4083baa2.tar.gz
cpython-8eef6a9ad04f6f81190f44ae3ded427e4083baa2.tar.bz2
Issue #22903: The fake test case created by unittest.loader when it fails importing a test module is now picklable.
Diffstat (limited to 'Lib/unittest')
-rw-r--r--Lib/unittest/loader.py31
-rw-r--r--Lib/unittest/test/test_discovery.py13
2 files changed, 34 insertions, 10 deletions
diff --git a/Lib/unittest/loader.py b/Lib/unittest/loader.py
index 8ee6c56..8a1a2a7 100644
--- a/Lib/unittest/loader.py
+++ b/Lib/unittest/loader.py
@@ -20,23 +20,34 @@ __unittest = True
VALID_MODULE_NAME = re.compile(r'[_a-z]\w*\.py$', re.IGNORECASE)
+class _FailedTest(case.TestCase):
+ _testMethodName = None
+
+ def __init__(self, method_name, exception):
+ self._exception = exception
+ super(_FailedTest, self).__init__(method_name)
+
+ def __getattr__(self, name):
+ if name != self._testMethodName:
+ return super(_FailedTest, self).__getattr__(name)
+ def testFailure():
+ raise self._exception
+ return testFailure
+
+
def _make_failed_import_test(name, suiteClass):
message = 'Failed to import test module: %s\n%s' % (
name, traceback.format_exc())
- return _make_failed_test('ModuleImportFailure', name, ImportError(message),
- suiteClass, message)
+ return _make_failed_test(name, ImportError(message), suiteClass, message)
def _make_failed_load_tests(name, exception, suiteClass):
message = 'Failed to call load_tests:\n%s' % (traceback.format_exc(),)
return _make_failed_test(
- 'LoadTestsFailure', name, exception, suiteClass, message)
+ name, exception, suiteClass, message)
-def _make_failed_test(classname, methodname, exception, suiteClass, message):
- def testFailure(self):
- raise exception
- attrs = {methodname: testFailure}
- TestClass = type(classname, (case.TestCase,), attrs)
- return suiteClass((TestClass(methodname),)), message
+def _make_failed_test(methodname, exception, suiteClass, message):
+ test = _FailedTest(methodname, exception)
+ return suiteClass((test,)), message
def _make_skipped_test(methodname, exception, suiteClass):
@case.skip(str(exception))
@@ -169,7 +180,7 @@ class TestLoader(object):
else:
# Otherwise, we signal that an AttributeError has occurred.
error_case, error_message = _make_failed_test(
- 'AttributeError', part, e, self.suiteClass,
+ part, e, self.suiteClass,
'Failed to access attribute:\n%s' % (
traceback.format_exc(),))
self.errors.append(error_message)
diff --git a/Lib/unittest/test/test_discovery.py b/Lib/unittest/test/test_discovery.py
index 4f61314..8991f38 100644
--- a/Lib/unittest/test/test_discovery.py
+++ b/Lib/unittest/test/test_discovery.py
@@ -3,6 +3,7 @@ from os.path import abspath
import re
import sys
import types
+import pickle
import builtins
from test import support
@@ -482,6 +483,10 @@ class TestDiscovery(unittest.TestCase):
test.my_package()
self.assertEqual(import_calls, ['my_package'])
+ # Check picklability
+ for proto in range(pickle.HIGHEST_PROTOCOL + 1):
+ pickle.loads(pickle.dumps(test, proto))
+
def test_discover_with_module_that_raises_SkipTest_on_import(self):
loader = unittest.TestLoader()
@@ -498,6 +503,10 @@ class TestDiscovery(unittest.TestCase):
suite.run(result)
self.assertEqual(len(result.skipped), 1)
+ # Check picklability
+ for proto in range(pickle.HIGHEST_PROTOCOL + 1):
+ pickle.loads(pickle.dumps(suite, proto))
+
def test_discover_with_init_module_that_raises_SkipTest_on_import(self):
vfs = {abspath('/foo'): ['my_package'],
abspath('/foo/my_package'): ['__init__.py', 'test_module.py']}
@@ -518,6 +527,10 @@ class TestDiscovery(unittest.TestCase):
self.assertEqual(result.testsRun, 1)
self.assertEqual(import_calls, ['my_package'])
+ # Check picklability
+ for proto in range(pickle.HIGHEST_PROTOCOL + 1):
+ pickle.loads(pickle.dumps(suite, proto))
+
def test_command_line_handling_parseArgs(self):
program = TestableTestProgram()