Fixed #16185, #15675 -- Added the ability for test runners to define custom options, and to specify a custom test runner at the command line. Thanks to Dmitry Jemerov and Mikołaj Siedlarek for the patch.
git-svn-id: http://code.djangoproject.com/svn/django/trunk@16352 bcc190cf-cafb-0310-a4f2-bffc1f526a37
This commit is contained in:
parent
b56ef75088
commit
046ffa483e
6
AUTHORS
6
AUTHORS
|
@ -252,6 +252,7 @@ answer newbie questions, and generally made Django that much better:
|
||||||
james_027@yahoo.com
|
james_027@yahoo.com
|
||||||
jcrasta@gmail.com
|
jcrasta@gmail.com
|
||||||
jdetaeye
|
jdetaeye
|
||||||
|
Dmitry Jemerov <intelliyole@gmail.com>
|
||||||
jhenry <jhenry@theonion.com>
|
jhenry <jhenry@theonion.com>
|
||||||
john@calixto.net
|
john@calixto.net
|
||||||
Zak Johnson <zakj@nox.cx>
|
Zak Johnson <zakj@nox.cx>
|
||||||
|
@ -338,6 +339,7 @@ answer newbie questions, and generally made Django that much better:
|
||||||
mark@junklight.com
|
mark@junklight.com
|
||||||
Orestis Markou <orestis@orestis.gr>
|
Orestis Markou <orestis@orestis.gr>
|
||||||
Takashi Matsuo <matsuo.takashi@gmail.com>
|
Takashi Matsuo <matsuo.takashi@gmail.com>
|
||||||
|
Zlatko Mašek <zlatko.masek@gmail.com>
|
||||||
Yasushi Masuda <whosaysni@gmail.com>
|
Yasushi Masuda <whosaysni@gmail.com>
|
||||||
mattycakes@gmail.com
|
mattycakes@gmail.com
|
||||||
Glenn Maynard <glenn@zewt.org>
|
Glenn Maynard <glenn@zewt.org>
|
||||||
|
@ -369,6 +371,7 @@ answer newbie questions, and generally made Django that much better:
|
||||||
Gopal Narayanan <gopastro@gmail.com>
|
Gopal Narayanan <gopastro@gmail.com>
|
||||||
Fraser Nevett <mail@nevett.org>
|
Fraser Nevett <mail@nevett.org>
|
||||||
Sam Newman <http://www.magpiebrain.com/>
|
Sam Newman <http://www.magpiebrain.com/>
|
||||||
|
Ryan Niemeyer <https://profiles.google.com/ryan.niemeyer/about>
|
||||||
Filip Noetzel <http://filip.noetzel.co.uk/>
|
Filip Noetzel <http://filip.noetzel.co.uk/>
|
||||||
Afonso Fernández Nogueira <fonzzo.django@gmail.com>
|
Afonso Fernández Nogueira <fonzzo.django@gmail.com>
|
||||||
Neal Norwitz <nnorwitz@google.com>
|
Neal Norwitz <nnorwitz@google.com>
|
||||||
|
@ -443,6 +446,7 @@ answer newbie questions, and generally made Django that much better:
|
||||||
Pete Shinners <pete@shinners.org>
|
Pete Shinners <pete@shinners.org>
|
||||||
Leo Shklovskii
|
Leo Shklovskii
|
||||||
jason.sidabras@gmail.com
|
jason.sidabras@gmail.com
|
||||||
|
Mikołaj Siedlarek <mikolaj.siedlarek@gmail.com>
|
||||||
Brenton Simpson <http://theillustratedlife.com>
|
Brenton Simpson <http://theillustratedlife.com>
|
||||||
Jozko Skrablin <jozko.skrablin@gmail.com>
|
Jozko Skrablin <jozko.skrablin@gmail.com>
|
||||||
Ben Slavin <benjamin.slavin@gmail.com>
|
Ben Slavin <benjamin.slavin@gmail.com>
|
||||||
|
@ -533,8 +537,6 @@ answer newbie questions, and generally made Django that much better:
|
||||||
Gasper Zejn <zejn@kiberpipa.org>
|
Gasper Zejn <zejn@kiberpipa.org>
|
||||||
Jarek Zgoda <jarek.zgoda@gmail.com>
|
Jarek Zgoda <jarek.zgoda@gmail.com>
|
||||||
Cheng Zhang
|
Cheng Zhang
|
||||||
Zlatko Mašek <zlatko.masek@gmail.com>
|
|
||||||
Ryan Niemeyer <https://profiles.google.com/ryan.niemeyer/about>
|
|
||||||
|
|
||||||
A big THANK YOU goes to:
|
A big THANK YOU goes to:
|
||||||
|
|
||||||
|
|
|
@ -1,29 +1,56 @@
|
||||||
|
from django.conf import settings
|
||||||
from django.core.management.base import BaseCommand
|
from django.core.management.base import BaseCommand
|
||||||
from optparse import make_option
|
from optparse import make_option, OptionParser
|
||||||
import sys
|
import sys
|
||||||
|
from django.test.utils import get_runner
|
||||||
|
|
||||||
class Command(BaseCommand):
|
class Command(BaseCommand):
|
||||||
option_list = BaseCommand.option_list + (
|
option_list = BaseCommand.option_list + (
|
||||||
make_option('--noinput', action='store_false', dest='interactive', default=True,
|
make_option('--noinput', action='store_false', dest='interactive', default=True,
|
||||||
help='Tells Django to NOT prompt the user for input of any kind.'),
|
help='Tells Django to NOT prompt the user for input of any kind.'),
|
||||||
make_option('--failfast', action='store_true', dest='failfast', default=False,
|
make_option('--failfast', action='store_true', dest='failfast', default=False,
|
||||||
help='Tells Django to stop running the test suite after first failed test.')
|
help='Tells Django to stop running the test suite after first failed test.'),
|
||||||
|
make_option('--testrunner', action='store', dest='testrunner',
|
||||||
|
help='Tells Django to use specified test runner class instead of the one '+
|
||||||
|
'specified by the TEST_RUNNER setting.')
|
||||||
)
|
)
|
||||||
help = 'Runs the test suite for the specified applications, or the entire site if no apps are specified.'
|
help = 'Runs the test suite for the specified applications, or the entire site if no apps are specified.'
|
||||||
args = '[appname ...]'
|
args = '[appname ...]'
|
||||||
|
|
||||||
requires_model_validation = False
|
requires_model_validation = False
|
||||||
|
|
||||||
|
def run_from_argv(self, argv):
|
||||||
|
"""
|
||||||
|
Pre-parse the command line to extract the value of the --testrunner
|
||||||
|
option. This allows a test runner to define additional command line
|
||||||
|
arguments.
|
||||||
|
"""
|
||||||
|
self.test_runner = None
|
||||||
|
option = '--testrunner='
|
||||||
|
for arg in argv[2:]:
|
||||||
|
if arg.startswith(option):
|
||||||
|
self.test_runner = arg[len(option):]
|
||||||
|
break
|
||||||
|
super(Command, self).run_from_argv(argv)
|
||||||
|
|
||||||
|
def create_parser(self, prog_name, subcommand):
|
||||||
|
test_runner_class = get_runner(settings, self.test_runner)
|
||||||
|
options = self.option_list + getattr(test_runner_class, 'option_list', ())
|
||||||
|
return OptionParser(prog=prog_name,
|
||||||
|
usage=self.usage(subcommand),
|
||||||
|
version=self.get_version(),
|
||||||
|
option_list=options)
|
||||||
|
|
||||||
def handle(self, *test_labels, **options):
|
def handle(self, *test_labels, **options):
|
||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
from django.test.utils import get_runner
|
from django.test.utils import get_runner
|
||||||
|
|
||||||
verbosity = int(options.get('verbosity', 1))
|
TestRunner = get_runner(settings, options.get('testrunner'))
|
||||||
interactive = options.get('interactive', True)
|
options['verbosity'] = int(options.get('verbosity', 1))
|
||||||
failfast = options.get('failfast', False)
|
options.setdefault('interactive', True)
|
||||||
TestRunner = get_runner(settings)
|
options.setdefault('failfast', False)
|
||||||
|
|
||||||
test_runner = TestRunner(verbosity=verbosity, interactive=interactive, failfast=failfast)
|
test_runner = TestRunner(**options)
|
||||||
failures = test_runner.run_tests(test_labels)
|
failures = test_runner.run_tests(test_labels)
|
||||||
|
|
||||||
if failures:
|
if failures:
|
||||||
|
|
|
@ -118,8 +118,11 @@ def restore_warnings_state(state):
|
||||||
warnings.filters = state[:]
|
warnings.filters = state[:]
|
||||||
|
|
||||||
|
|
||||||
def get_runner(settings):
|
def get_runner(settings, test_runner_class=None):
|
||||||
test_path = settings.TEST_RUNNER.split('.')
|
if not test_runner_class:
|
||||||
|
test_runner_class = settings.TEST_RUNNER
|
||||||
|
|
||||||
|
test_path = test_runner_class.split('.')
|
||||||
# Allow for Python 2.5 relative paths
|
# Allow for Python 2.5 relative paths
|
||||||
if len(test_path) > 1:
|
if len(test_path) > 1:
|
||||||
test_module_name = '.'.join(test_path[:-1])
|
test_module_name = '.'.join(test_path[:-1])
|
||||||
|
|
|
@ -964,6 +964,13 @@ information.
|
||||||
Use the :djadminopt:`--failfast` option to stop running tests and report the failure
|
Use the :djadminopt:`--failfast` option to stop running tests and report the failure
|
||||||
immediately after a test fails.
|
immediately after a test fails.
|
||||||
|
|
||||||
|
.. versionadded:: 1.4
|
||||||
|
.. django-admin-option:: --testrunner
|
||||||
|
|
||||||
|
The :djandminopt:`--testrunner` option can be used to control the test runner
|
||||||
|
class that is used to execute tests. If this value is provided, it overrides
|
||||||
|
the value provided by the :setting:`TEST_RUNNER` setting.
|
||||||
|
|
||||||
testserver <fixture fixture ...>
|
testserver <fixture fixture ...>
|
||||||
--------------------------------
|
--------------------------------
|
||||||
|
|
||||||
|
|
|
@ -1744,6 +1744,29 @@ set up, execute and tear down the test suite.
|
||||||
write your own test runner, ensure accept and handle the ``**kwargs``
|
write your own test runner, ensure accept and handle the ``**kwargs``
|
||||||
parameter.
|
parameter.
|
||||||
|
|
||||||
|
.. versionadded:: 1.4
|
||||||
|
|
||||||
|
Your test runner may also define additional command-line options.
|
||||||
|
If you add an ``option_list`` attribute to a subclassed test runner,
|
||||||
|
those options will be added to the list of command-line options that
|
||||||
|
the :djadmin:`test` command can use.
|
||||||
|
|
||||||
|
|
||||||
|
Attributes
|
||||||
|
~~~~~~~~~~
|
||||||
|
|
||||||
|
|
||||||
|
.. attribute:: DjangoTestSuiteRunner.option_list
|
||||||
|
|
||||||
|
.. versionadded:: 1.4
|
||||||
|
|
||||||
|
This is the tuple of ``optparse`` options which will be fed into the
|
||||||
|
management command's ``OptionParser`` for parsing arguments. See the
|
||||||
|
documentation for Python's ``optparse`` module for more details.
|
||||||
|
|
||||||
|
Methods
|
||||||
|
~~~~~~~
|
||||||
|
|
||||||
.. method:: DjangoTestSuiteRunner.run_tests(test_labels, extra_tests=None, **kwargs)
|
.. method:: DjangoTestSuiteRunner.run_tests(test_labels, extra_tests=None, **kwargs)
|
||||||
|
|
||||||
Run the test suite.
|
Run the test suite.
|
||||||
|
|
|
@ -2,12 +2,15 @@
|
||||||
Tests for django test runner
|
Tests for django test runner
|
||||||
"""
|
"""
|
||||||
import StringIO
|
import StringIO
|
||||||
|
from optparse import make_option
|
||||||
import warnings
|
import warnings
|
||||||
|
|
||||||
from django.core.exceptions import ImproperlyConfigured
|
from django.core.exceptions import ImproperlyConfigured
|
||||||
|
from django.core.management import call_command
|
||||||
from django.test import simple
|
from django.test import simple
|
||||||
from django.test.utils import get_warnings_state, restore_warnings_state
|
from django.test.utils import get_warnings_state, restore_warnings_state
|
||||||
from django.utils import unittest
|
from django.utils import unittest
|
||||||
|
from regressiontests.admin_scripts.tests import AdminScriptTestCase
|
||||||
|
|
||||||
|
|
||||||
class DjangoTestRunnerTests(unittest.TestCase):
|
class DjangoTestRunnerTests(unittest.TestCase):
|
||||||
|
@ -128,3 +131,75 @@ class DependencyOrderingTests(unittest.TestCase):
|
||||||
|
|
||||||
self.assertRaises(ImproperlyConfigured, simple.dependency_ordered, raw, dependencies=dependencies)
|
self.assertRaises(ImproperlyConfigured, simple.dependency_ordered, raw, dependencies=dependencies)
|
||||||
|
|
||||||
|
|
||||||
|
class MockTestRunner(object):
|
||||||
|
invoked = False
|
||||||
|
|
||||||
|
def __init__(self, *args, **kwargs):
|
||||||
|
pass
|
||||||
|
|
||||||
|
def run_tests(self, test_labels, extra_tests=None, **kwargs):
|
||||||
|
MockTestRunner.invoked = True
|
||||||
|
|
||||||
|
|
||||||
|
class ManageCommandTests(unittest.TestCase):
|
||||||
|
|
||||||
|
def test_custom_test_runner(self):
|
||||||
|
call_command('test', 'sites',
|
||||||
|
testrunner='regressiontests.test_runner.tests.MockTestRunner')
|
||||||
|
self.assertTrue(MockTestRunner.invoked,
|
||||||
|
"The custom test runner has not been invoked")
|
||||||
|
|
||||||
|
|
||||||
|
class CustomOptionsTestRunner(simple.DjangoTestSuiteRunner):
|
||||||
|
option_list = (
|
||||||
|
make_option('--option_a','-a', action='store', dest='option_a', default='1'),
|
||||||
|
make_option('--option_b','-b', action='store', dest='option_b', default='2'),
|
||||||
|
make_option('--option_c','-c', action='store', dest='option_c', default='3'),
|
||||||
|
)
|
||||||
|
|
||||||
|
def __init__(self, verbosity=1, interactive=True, failfast=True, option_a=None, option_b=None, option_c=None, **kwargs):
|
||||||
|
super(CustomOptionsTestRunner, self).__init__(verbosity=verbosity, interactive=interactive,
|
||||||
|
failfast=failfast)
|
||||||
|
self.option_a = option_a
|
||||||
|
self.option_b = option_b
|
||||||
|
self.option_c = option_c
|
||||||
|
|
||||||
|
def run_tests(self, test_labels, extra_tests=None, **kwargs):
|
||||||
|
print "%s:%s:%s" % (self.option_a, self.option_b, self.option_c)
|
||||||
|
|
||||||
|
|
||||||
|
class CustomTestRunnerOptionsTests(AdminScriptTestCase):
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
settings = {
|
||||||
|
'TEST_RUNNER': '\'regressiontests.test_runner.tests.CustomOptionsTestRunner\'',
|
||||||
|
}
|
||||||
|
self.write_settings('settings.py', sdict=settings)
|
||||||
|
|
||||||
|
def tearDown(self):
|
||||||
|
self.remove_settings('settings.py')
|
||||||
|
|
||||||
|
def test_default_options(self):
|
||||||
|
args = ['test', '--settings=settings']
|
||||||
|
out, err = self.run_django_admin(args)
|
||||||
|
self.assertNoOutput(err)
|
||||||
|
self.assertOutput(out, '1:2:3')
|
||||||
|
|
||||||
|
def test_default_and_given_options(self):
|
||||||
|
args = ['test', '--settings=settings', '--option_b=foo']
|
||||||
|
out, err = self.run_django_admin(args)
|
||||||
|
self.assertNoOutput(err)
|
||||||
|
self.assertOutput(out, '1:foo:3')
|
||||||
|
|
||||||
|
def test_option_name_and_value_separated(self):
|
||||||
|
args = ['test', '--settings=settings', '--option_b', 'foo']
|
||||||
|
out, err = self.run_django_admin(args)
|
||||||
|
self.assertNoOutput(err)
|
||||||
|
self.assertOutput(out, '1:foo:3')
|
||||||
|
|
||||||
|
def test_all_options_given(self):
|
||||||
|
args = ['test', '--settings=settings', '--option_a=bar', '--option_b=foo', '--option_c=31337']
|
||||||
|
out, err = self.run_django_admin(args)
|
||||||
|
self.assertNoOutput(err)
|
||||||
|
self.assertOutput(out, 'bar:foo:31337')
|
||||||
|
|
Loading…
Reference in New Issue