2010-01-01 02:48:28 +08:00
|
|
|
import sys
|
|
|
|
import signal
|
2007-05-31 21:18:12 +08:00
|
|
|
import unittest
|
2010-01-01 02:48:28 +08:00
|
|
|
|
2006-08-27 20:24:59 +08:00
|
|
|
from django.conf import settings
|
2007-07-28 12:02:52 +08:00
|
|
|
from django.db.models import get_app, get_apps
|
2007-05-31 21:18:12 +08:00
|
|
|
from django.test import _doctest as doctest
|
2006-09-02 17:26:24 +08:00
|
|
|
from django.test.utils import setup_test_environment, teardown_test_environment
|
2009-01-16 10:30:22 +08:00
|
|
|
from django.test.testcases import OutputChecker, DocTestRunner, TestCase
|
2006-08-27 20:24:59 +08:00
|
|
|
|
|
|
|
# The module name for tests outside models.py
|
|
|
|
TEST_MODULE = 'tests'
|
2009-02-22 16:34:51 +08:00
|
|
|
|
2006-08-27 20:24:59 +08:00
|
|
|
doctestOutputChecker = OutputChecker()
|
|
|
|
|
2009-12-14 00:24:36 +08:00
|
|
|
class DjangoTestRunner(unittest.TextTestRunner):
|
2010-01-01 02:48:28 +08:00
|
|
|
|
2009-12-14 00:24:36 +08:00
|
|
|
def __init__(self, verbosity=0, failfast=False, **kwargs):
|
|
|
|
super(DjangoTestRunner, self).__init__(verbosity=verbosity, **kwargs)
|
|
|
|
self.failfast = failfast
|
2010-01-01 02:48:28 +08:00
|
|
|
self._keyboard_interrupt_intercepted = False
|
|
|
|
|
|
|
|
def run(self, *args, **kwargs):
|
|
|
|
"""
|
|
|
|
Runs the test suite after registering a custom signal handler
|
|
|
|
that triggers a graceful exit when Ctrl-C is pressed.
|
|
|
|
"""
|
|
|
|
self._default_keyboard_interrupt_handler = signal.signal(signal.SIGINT,
|
|
|
|
self._keyboard_interrupt_handler)
|
|
|
|
result = super(DjangoTestRunner, self).run(*args, **kwargs)
|
|
|
|
signal.signal(signal.SIGINT, self._default_keyboard_interrupt_handler)
|
|
|
|
return result
|
|
|
|
|
|
|
|
def _keyboard_interrupt_handler(self, signal_number, stack_frame):
|
|
|
|
"""
|
|
|
|
Handles Ctrl-C by setting a flag that will stop the test run when
|
|
|
|
the currently running test completes.
|
|
|
|
"""
|
|
|
|
self._keyboard_interrupt_intercepted = True
|
|
|
|
sys.stderr.write(" <Test run halted by Ctrl-C> ")
|
|
|
|
# Set the interrupt handler back to the default handler, so that
|
|
|
|
# another Ctrl-C press will trigger immediate exit.
|
|
|
|
signal.signal(signal.SIGINT, self._default_keyboard_interrupt_handler)
|
|
|
|
|
2009-12-14 00:24:36 +08:00
|
|
|
def _makeResult(self):
|
|
|
|
result = super(DjangoTestRunner, self)._makeResult()
|
|
|
|
failfast = self.failfast
|
2010-01-01 02:48:28 +08:00
|
|
|
|
2009-12-14 00:24:36 +08:00
|
|
|
def stoptest_override(func):
|
|
|
|
def stoptest(test):
|
2010-01-01 02:48:28 +08:00
|
|
|
# If we were set to failfast and the unit test failed,
|
|
|
|
# or if the user has typed Ctrl-C, report and quit
|
|
|
|
if (failfast and not result.wasSuccessful()) or \
|
|
|
|
self._keyboard_interrupt_intercepted:
|
2009-12-14 00:24:36 +08:00
|
|
|
result.stop()
|
|
|
|
func(test)
|
|
|
|
return stoptest
|
2010-01-01 02:48:28 +08:00
|
|
|
|
2009-12-14 00:24:36 +08:00
|
|
|
setattr(result, 'stopTest', stoptest_override(result.stopTest))
|
|
|
|
return result
|
|
|
|
|
2007-07-28 12:02:52 +08:00
|
|
|
def get_tests(app_module):
|
|
|
|
try:
|
|
|
|
app_path = app_module.__name__.split('.')[:-1]
|
|
|
|
test_module = __import__('.'.join(app_path + [TEST_MODULE]), {}, {}, TEST_MODULE)
|
|
|
|
except ImportError, e:
|
|
|
|
# Couldn't import tests.py. Was it due to a missing file, or
|
|
|
|
# due to an import error in a tests.py that actually exists?
|
|
|
|
import os.path
|
|
|
|
from imp import find_module
|
|
|
|
try:
|
|
|
|
mod = find_module(TEST_MODULE, [os.path.dirname(app_module.__file__)])
|
|
|
|
except ImportError:
|
|
|
|
# 'tests' module doesn't exist. Move on.
|
|
|
|
test_module = None
|
|
|
|
else:
|
2009-02-22 16:34:51 +08:00
|
|
|
# The module exists, so there must be an import error in the
|
2007-07-28 12:02:52 +08:00
|
|
|
# test module itself. We don't need the module; so if the
|
|
|
|
# module was a single file module (i.e., tests.py), close the file
|
|
|
|
# handle returned by find_module. Otherwise, the test module
|
|
|
|
# is a directory, and there is nothing to close.
|
|
|
|
if mod[0]:
|
|
|
|
mod[0].close()
|
|
|
|
raise
|
|
|
|
return test_module
|
2009-02-22 16:34:51 +08:00
|
|
|
|
2006-08-27 20:24:59 +08:00
|
|
|
def build_suite(app_module):
|
|
|
|
"Create a complete Django test suite for the provided application module"
|
|
|
|
suite = unittest.TestSuite()
|
2009-02-22 16:34:51 +08:00
|
|
|
|
2007-07-20 21:57:49 +08:00
|
|
|
# Load unit and doctests in the models.py module. If module has
|
|
|
|
# a suite() method, use it. Otherwise build the test suite ourselves.
|
|
|
|
if hasattr(app_module, 'suite'):
|
|
|
|
suite.addTest(app_module.suite())
|
|
|
|
else:
|
|
|
|
suite.addTest(unittest.defaultTestLoader.loadTestsFromModule(app_module))
|
|
|
|
try:
|
|
|
|
suite.addTest(doctest.DocTestSuite(app_module,
|
|
|
|
checker=doctestOutputChecker,
|
|
|
|
runner=DocTestRunner))
|
|
|
|
except ValueError:
|
|
|
|
# No doc tests in models.py
|
|
|
|
pass
|
2009-02-22 16:34:51 +08:00
|
|
|
|
|
|
|
# Check to see if a separate 'tests' module exists parallel to the
|
2006-08-27 20:24:59 +08:00
|
|
|
# models module
|
2007-07-28 12:02:52 +08:00
|
|
|
test_module = get_tests(app_module)
|
|
|
|
if test_module:
|
2007-07-20 21:57:49 +08:00
|
|
|
# Load unit and doctests in the tests.py module. If module has
|
|
|
|
# a suite() method, use it. Otherwise build the test suite ourselves.
|
|
|
|
if hasattr(test_module, 'suite'):
|
|
|
|
suite.addTest(test_module.suite())
|
|
|
|
else:
|
|
|
|
suite.addTest(unittest.defaultTestLoader.loadTestsFromModule(test_module))
|
2009-02-22 16:34:51 +08:00
|
|
|
try:
|
|
|
|
suite.addTest(doctest.DocTestSuite(test_module,
|
2007-07-20 21:57:49 +08:00
|
|
|
checker=doctestOutputChecker,
|
|
|
|
runner=DocTestRunner))
|
|
|
|
except ValueError:
|
|
|
|
# No doc tests in tests.py
|
|
|
|
pass
|
2006-08-27 20:24:59 +08:00
|
|
|
return suite
|
|
|
|
|
2007-07-28 12:02:52 +08:00
|
|
|
def build_test(label):
|
2009-02-22 16:34:51 +08:00
|
|
|
"""Construct a test case a test with the specified label. Label should
|
2007-07-28 12:02:52 +08:00
|
|
|
be of the form model.TestClass or model.TestClass.test_method. Returns
|
|
|
|
an instantiated test or test suite corresponding to the label provided.
|
2009-02-22 16:34:51 +08:00
|
|
|
|
2007-07-28 12:02:52 +08:00
|
|
|
"""
|
|
|
|
parts = label.split('.')
|
|
|
|
if len(parts) < 2 or len(parts) > 3:
|
|
|
|
raise ValueError("Test label '%s' should be of the form app.TestCase or app.TestCase.test_method" % label)
|
2009-02-22 16:34:51 +08:00
|
|
|
|
2007-07-28 12:02:52 +08:00
|
|
|
app_module = get_app(parts[0])
|
|
|
|
TestClass = getattr(app_module, parts[1], None)
|
|
|
|
|
|
|
|
# Couldn't find the test class in models.py; look in tests.py
|
|
|
|
if TestClass is None:
|
|
|
|
test_module = get_tests(app_module)
|
|
|
|
if test_module:
|
|
|
|
TestClass = getattr(test_module, parts[1], None)
|
|
|
|
|
|
|
|
if len(parts) == 2: # label is app.TestClass
|
|
|
|
try:
|
|
|
|
return unittest.TestLoader().loadTestsFromTestCase(TestClass)
|
|
|
|
except TypeError:
|
2009-02-22 16:34:51 +08:00
|
|
|
raise ValueError("Test label '%s' does not refer to a test class" % label)
|
2007-07-28 12:02:52 +08:00
|
|
|
else: # label is app.TestClass.test_method
|
2009-02-22 16:34:51 +08:00
|
|
|
if not TestClass:
|
|
|
|
raise ValueError("Test label '%s' does not refer to a test class" % label)
|
2007-07-28 12:02:52 +08:00
|
|
|
return TestClass(parts[2])
|
|
|
|
|
2009-01-18 06:04:40 +08:00
|
|
|
# Python 2.3 compatibility: TestSuites were made iterable in 2.4.
|
|
|
|
# We need to iterate over them, so we add the missing method when
|
2009-02-22 16:34:51 +08:00
|
|
|
# necessary.
|
2009-01-18 06:04:40 +08:00
|
|
|
try:
|
|
|
|
getattr(unittest.TestSuite, '__iter__')
|
|
|
|
except AttributeError:
|
|
|
|
setattr(unittest.TestSuite, '__iter__', lambda s: iter(s._tests))
|
|
|
|
|
2009-01-16 10:30:22 +08:00
|
|
|
def partition_suite(suite, classes, bins):
|
|
|
|
"""
|
|
|
|
Partitions a test suite by test type.
|
2009-02-22 16:34:51 +08:00
|
|
|
|
2009-01-16 10:30:22 +08:00
|
|
|
classes is a sequence of types
|
|
|
|
bins is a sequence of TestSuites, one more than classes
|
2009-02-22 16:34:51 +08:00
|
|
|
|
|
|
|
Tests of type classes[i] are added to bins[i],
|
2009-01-16 10:30:22 +08:00
|
|
|
tests with no match found in classes are place in bins[-1]
|
|
|
|
"""
|
|
|
|
for test in suite:
|
|
|
|
if isinstance(test, unittest.TestSuite):
|
|
|
|
partition_suite(test, classes, bins)
|
|
|
|
else:
|
|
|
|
for i in range(len(classes)):
|
|
|
|
if isinstance(test, classes[i]):
|
|
|
|
bins[i].addTest(test)
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
bins[-1].addTest(test)
|
2009-02-22 16:34:51 +08:00
|
|
|
|
2009-01-16 10:30:22 +08:00
|
|
|
def reorder_suite(suite, classes):
|
|
|
|
"""
|
|
|
|
Reorders a test suite by test type.
|
2009-02-22 16:34:51 +08:00
|
|
|
|
2009-01-16 10:30:22 +08:00
|
|
|
classes is a sequence of types
|
2009-02-22 16:34:51 +08:00
|
|
|
|
2009-01-16 10:30:22 +08:00
|
|
|
All tests of type clases[0] are placed first, then tests of type classes[1], etc.
|
|
|
|
Tests with no match in classes are placed last.
|
|
|
|
"""
|
|
|
|
class_count = len(classes)
|
|
|
|
bins = [unittest.TestSuite() for i in range(class_count+1)]
|
|
|
|
partition_suite(suite, classes, bins)
|
|
|
|
for i in range(class_count):
|
|
|
|
bins[0].addTests(bins[i+1])
|
|
|
|
return bins[0]
|
|
|
|
|
2009-12-14 00:24:36 +08:00
|
|
|
def run_tests(test_labels, verbosity=1, interactive=True, failfast=False, extra_tests=[]):
|
2006-08-27 20:24:59 +08:00
|
|
|
"""
|
2007-07-28 12:02:52 +08:00
|
|
|
Run the unit tests for all the test labels in the provided list.
|
|
|
|
Labels must be of the form:
|
|
|
|
- app.TestClass.test_method
|
|
|
|
Run a single specific test method
|
|
|
|
- app.TestClass
|
|
|
|
Run all the test methods in a given class
|
|
|
|
- app
|
|
|
|
Search for doctests and unittests in the named application.
|
|
|
|
|
|
|
|
When looking for tests, the test runner will look in the models and
|
|
|
|
tests modules for the application.
|
2009-02-22 16:34:51 +08:00
|
|
|
|
2007-07-28 12:02:52 +08:00
|
|
|
A list of 'extra' tests may also be provided; these tests
|
2006-08-27 20:24:59 +08:00
|
|
|
will be added to the test suite.
|
2009-02-22 16:34:51 +08:00
|
|
|
|
2007-02-26 20:52:01 +08:00
|
|
|
Returns the number of tests that failed.
|
2006-08-27 20:24:59 +08:00
|
|
|
"""
|
2006-09-02 17:26:24 +08:00
|
|
|
setup_test_environment()
|
2009-02-22 16:34:51 +08:00
|
|
|
|
|
|
|
settings.DEBUG = False
|
2006-08-27 20:24:59 +08:00
|
|
|
suite = unittest.TestSuite()
|
2009-02-22 16:34:51 +08:00
|
|
|
|
2007-07-28 12:02:52 +08:00
|
|
|
if test_labels:
|
|
|
|
for label in test_labels:
|
|
|
|
if '.' in label:
|
|
|
|
suite.addTest(build_test(label))
|
|
|
|
else:
|
|
|
|
app = get_app(label)
|
|
|
|
suite.addTest(build_suite(app))
|
|
|
|
else:
|
|
|
|
for app in get_apps():
|
|
|
|
suite.addTest(build_suite(app))
|
2009-02-22 16:34:51 +08:00
|
|
|
|
2006-08-27 20:24:59 +08:00
|
|
|
for test in extra_tests:
|
|
|
|
suite.addTest(test)
|
|
|
|
|
2009-01-16 10:30:22 +08:00
|
|
|
suite = reorder_suite(suite, (TestCase,))
|
|
|
|
|
2009-12-22 23:18:51 +08:00
|
|
|
from django.db import connections
|
|
|
|
old_names = []
|
|
|
|
for alias in connections:
|
|
|
|
connection = connections[alias]
|
|
|
|
old_names.append((connection, connection.settings_dict['NAME']))
|
|
|
|
connection.creation.create_test_db(verbosity, autoclobber=not interactive)
|
2009-12-14 00:24:36 +08:00
|
|
|
result = DjangoTestRunner(verbosity=verbosity, failfast=failfast).run(suite)
|
2009-12-22 23:18:51 +08:00
|
|
|
for connection, old_name in old_names:
|
|
|
|
connection.creation.destroy_test_db(old_name, verbosity)
|
2009-02-22 16:34:51 +08:00
|
|
|
|
2006-09-02 17:26:24 +08:00
|
|
|
teardown_test_environment()
|
2009-02-22 16:34:51 +08:00
|
|
|
|
2007-03-29 19:59:31 +08:00
|
|
|
return len(result.failures) + len(result.errors)
|