Commit 3f5f824b authored by Stefan Behnel's avatar Stefan Behnel

enable forked test runs for pyregr test suite

parent 31d50eb9
...@@ -588,64 +588,70 @@ class CythonRunTestCase(CythonCompileTestCase): ...@@ -588,64 +588,70 @@ class CythonRunTestCase(CythonCompileTestCase):
self.run_doctests(self.module, result) self.run_doctests(self.module, result)
def run_doctests(self, module_name, result): def run_doctests(self, module_name, result):
if sys.version_info[0] >= 3 or not hasattr(os, 'fork') or not self.fork: def run_test(result):
doctest.DocTestSuite(module_name).run(result) tests = doctest.DocTestSuite(module_name)
gc.collect() tests.run(result)
return run_forked_test(result, run_test, self.shortDescription(), self.fork)
# fork to make sure we do not keep the tested module loaded
result_handle, result_file = tempfile.mkstemp()
os.close(result_handle)
child_id = os.fork()
if not child_id:
result_code = 0
try:
try:
tests = None
try:
partial_result = PartialTestResult(result)
tests = doctest.DocTestSuite(module_name)
tests.run(partial_result)
gc.collect()
except Exception:
if tests is None:
# importing failed, try to fake a test class
tests = _FakeClass(
failureException=sys.exc_info()[1],
_shortDescription=self.shortDescription(),
module_name=None)
partial_result.addError(tests, sys.exc_info())
result_code = 1
output = open(result_file, 'wb')
pickle.dump(partial_result.data(), output)
except:
traceback.print_exc()
finally:
try: output.close()
except: pass
os._exit(result_code)
def run_forked_test(result, run_func, test_name, fork=True):
if sys.version_info[0] >= 3 or not hasattr(os, 'fork') or not fork:
run_test(result)
gc.collect()
return
# fork to make sure we do not keep the tested module loaded
result_handle, result_file = tempfile.mkstemp()
os.close(result_handle)
child_id = os.fork()
if not child_id:
result_code = 0
try: try:
cid, result_code = os.waitpid(child_id, 0) try:
# os.waitpid returns the child's result code in the tests = None
# upper byte of result_code, and the signal it was
# killed by in the lower byte
if result_code & 255:
raise Exception("Tests in module '%s' were unexpectedly killed by signal %d"%
(module_name, result_code & 255))
result_code = result_code >> 8
if result_code in (0,1):
input = open(result_file, 'rb')
try: try:
PartialTestResult.join_results(result, pickle.load(input)) partial_result = PartialTestResult(result)
finally: run_func(partial_result)
input.close() gc.collect()
if result_code: except Exception:
raise Exception("Tests in module '%s' exited with status %d" % if tests is None:
(module_name, result_code)) # importing failed, try to fake a test class
tests = _FakeClass(
failureException=sys.exc_info()[1],
_shortDescription=test_name,
module_name=None)
partial_result.addError(tests, sys.exc_info())
result_code = 1
output = open(result_file, 'wb')
pickle.dump(partial_result.data(), output)
except:
traceback.print_exc()
finally: finally:
try: os.unlink(result_file) try: output.close()
except: pass except: pass
os._exit(result_code)
try:
cid, result_code = os.waitpid(child_id, 0)
# os.waitpid returns the child's result code in the
# upper byte of result_code, and the signal it was
# killed by in the lower byte
if result_code & 255:
raise Exception("Tests in module '%s' were unexpectedly killed by signal %d"%
(module_name, result_code & 255))
result_code = result_code >> 8
if result_code in (0,1):
input = open(result_file, 'rb')
try:
PartialTestResult.join_results(result, pickle.load(input))
finally:
input.close()
if result_code:
raise Exception("Tests in module '%s' exited with status %d" %
(module_name, result_code))
finally:
try: os.unlink(result_file)
except: pass
class PureDoctestTestCase(unittest.TestCase): class PureDoctestTestCase(unittest.TestCase):
def __init__(self, module_name, module_path): def __init__(self, module_name, module_path):
...@@ -773,20 +779,23 @@ class CythonPyregrTestCase(CythonRunTestCase): ...@@ -773,20 +779,23 @@ class CythonPyregrTestCase(CythonRunTestCase):
except ImportError: # Py3k except ImportError: # Py3k
from test import support from test import support
def run_unittest(*classes): def run_test(result):
return self._run_unittest(result, *classes) def run_unittest(*classes):
def run_doctest(module, verbosity=None): return self._run_unittest(result, *classes)
return self._run_doctest(result, module) def run_doctest(module, verbosity=None):
return self._run_doctest(result, module)
support.run_unittest = run_unittest support.run_unittest = run_unittest
support.run_doctest = run_doctest support.run_doctest = run_doctest
try: try:
module = __import__(self.module) module = __import__(self.module)
if hasattr(module, 'test_main'): if hasattr(module, 'test_main'):
module.test_main() module.test_main()
except (unittest.SkipTest, support.ResourceDenied): except (unittest.SkipTest, support.ResourceDenied):
result.addSkip(self, 'ok') result.addSkip(self, 'ok')
run_forked_test(result, run_test, self.shortDescription(), self.fork)
include_debugger = sys.version_info[:2] > (2, 5) include_debugger = sys.version_info[:2] > (2, 5)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment