Converted test management command to argparse

Keeping backwards compatibility with test_runner.option_list is
tricky and would imply transforming an optparse.Option to an
argparse.Action. I choose to introduce a backwards incompatible
change because it only affects testing, not runtime behavior.
This commit is contained in:
Claude Paroz 2014-06-06 20:12:23 +02:00
parent cbff097bd9
commit 4b4524291a
5 changed files with 86 additions and 50 deletions

View File

@ -1,7 +1,6 @@
import logging import logging
import sys import sys
import os import os
from optparse import make_option, OptionParser
from django.conf import settings from django.conf import settings
from django.core.management.base import BaseCommand from django.core.management.base import BaseCommand
@ -9,26 +8,7 @@ from django.test.utils import get_runner
class Command(BaseCommand): class Command(BaseCommand):
option_list = BaseCommand.option_list + ( help = 'Discover and run tests in the specified modules or the current directory.'
make_option('--noinput',
action='store_false', dest='interactive', default=True,
help='Tells Django to NOT prompt the user for input of any kind.'),
make_option('--failfast',
action='store_true', dest='failfast', default=False,
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.'),
make_option('--liveserver',
action='store', dest='liveserver', default=None,
help='Overrides the default address where the live server (used '
'with LiveServerTestCase) is expected to run from. The '
'default value is localhost:8081.'),
)
help = ('Discover and run tests in the specified modules or the current directory.')
args = '[path.to.modulename|path.to.modulename.TestCase|path.to.modulename.TestCase.test_method]...'
requires_system_checks = False requires_system_checks = False
@ -49,15 +29,40 @@ class Command(BaseCommand):
break break
super(Command, self).run_from_argv(argv) super(Command, self).run_from_argv(argv)
def create_parser(self, prog_name, subcommand): def add_arguments(self, parser):
parser = super(Command, self).create_parser(prog_name, subcommand) parser.add_argument('args', metavar='test_label', nargs='*',
help='Module paths to test; can be modulename, modulename.TestCase or modulename.TestCase.test_method')
parser.add_argument('--noinput',
action='store_false', dest='interactive', default=True,
help='Tells Django to NOT prompt the user for input of any kind.'),
parser.add_argument('--failfast',
action='store_true', dest='failfast', default=False,
help='Tells Django to stop running the test suite after first '
'failed test.'),
parser.add_argument('--testrunner',
action='store', dest='testrunner',
help='Tells Django to use specified test runner class instead of '
'the one specified by the TEST_RUNNER setting.'),
parser.add_argument('--liveserver',
action='store', dest='liveserver', default=None,
help='Overrides the default address where the live server (used '
'with LiveServerTestCase) is expected to run from. The '
'default value is localhost:8081.'),
test_runner_class = get_runner(settings, self.test_runner) test_runner_class = get_runner(settings, self.test_runner)
for opt in getattr(test_runner_class, 'option_list', ()): if hasattr(test_runner_class, 'option_list'):
parser.add_option(opt) # Keeping compatibility with both optparse and argparse at this level
return parser # would be too heavy for a non-critical item
raise RuntimeError(
"The method to extend accepted command-line arguments by the "
"test management command has changed in Django 1.8. Please "
"create an add_arguments class method to achieve this.")
if hasattr(test_runner_class, 'add_arguments'):
test_runner_class.add_arguments(parser)
def execute(self, *args, **options): def execute(self, *args, **options):
if int(options['verbosity']) > 0: if options['verbosity'] > 0:
# ensure that deprecation warnings are displayed during testing # ensure that deprecation warnings are displayed during testing
# the following state is assumed: # the following state is assumed:
# logging.capturewarnings is true # logging.capturewarnings is true
@ -67,7 +72,7 @@ class Command(BaseCommand):
handler = logging.StreamHandler() handler = logging.StreamHandler()
logger.addHandler(handler) logger.addHandler(handler)
super(Command, self).execute(*args, **options) super(Command, self).execute(*args, **options)
if int(options['verbosity']) > 0: if options['verbosity'] > 0:
# remove the testing-specific handler # remove the testing-specific handler
logger.removeHandler(handler) logger.removeHandler(handler)
@ -76,7 +81,6 @@ class Command(BaseCommand):
from django.test.utils import get_runner from django.test.utils import get_runner
TestRunner = get_runner(settings, options.get('testrunner')) TestRunner = get_runner(settings, options.get('testrunner'))
options['verbosity'] = int(options.get('verbosity'))
if options.get('liveserver') is not None: if options.get('liveserver') is not None:
os.environ['DJANGO_LIVE_TEST_SERVER_ADDRESS'] = options['liveserver'] os.environ['DJANGO_LIVE_TEST_SERVER_ADDRESS'] = options['liveserver']

View File

@ -1,6 +1,5 @@
from importlib import import_module from importlib import import_module
import os import os
from optparse import make_option
import unittest import unittest
from unittest import TestSuite, defaultTestLoader from unittest import TestSuite, defaultTestLoader
@ -19,17 +18,6 @@ class DiscoverRunner(object):
test_runner = unittest.TextTestRunner test_runner = unittest.TextTestRunner
test_loader = defaultTestLoader test_loader = defaultTestLoader
reorder_by = (TestCase, SimpleTestCase) reorder_by = (TestCase, SimpleTestCase)
option_list = (
make_option('-t', '--top-level-directory',
action='store', dest='top_level', default=None,
help='Top level of project for unittest discovery.'),
make_option('-p', '--pattern', action='store', dest='pattern',
default="test*.py",
help='The test matching pattern. Defaults to test*.py.'),
make_option('-k', '--keepdb', action='store_true', dest='keepdb',
default=False,
help='Preserve the test DB between runs. Defaults to False'),
)
def __init__(self, pattern=None, top_level=None, def __init__(self, pattern=None, top_level=None,
verbosity=1, interactive=True, failfast=False, keepdb=False, verbosity=1, interactive=True, failfast=False, keepdb=False,
@ -43,6 +31,18 @@ class DiscoverRunner(object):
self.failfast = failfast self.failfast = failfast
self.keepdb = keepdb self.keepdb = keepdb
@classmethod
def add_arguments(cls, parser):
parser.add_argument('-t', '--top-level-directory',
action='store', dest='top_level', default=None,
help='Top level of project for unittest discovery.')
parser.add_argument('-p', '--pattern', action='store', dest='pattern',
default="test*.py",
help='The test matching pattern. Defaults to test*.py.')
parser.add_argument('-k', '--keepdb', action='store_true', dest='keepdb',
default=False,
help='Preserve the test DB between runs. Defaults to False')
def setup_test_environment(self, **kwargs): def setup_test_environment(self, **kwargs):
setup_test_environment() setup_test_environment()
settings.DEBUG = False settings.DEBUG = False

View File

@ -294,6 +294,17 @@ you don't have to keep compatibility with older Django versions, it's better to
implement the new :meth:`~django.core.management.BaseCommand.add_arguments` implement the new :meth:`~django.core.management.BaseCommand.add_arguments`
method as described in :doc:`/howto/custom-management-commands`. method as described in :doc:`/howto/custom-management-commands`.
Custom test management command arguments through test runner
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
The method to add custom arguments to the `test` management command through the
test runner has changed. Previously, you could provide an `option_list` class
variable on the test runner to add more arguments (à la :py:mod:`optparse`).
Now to implement the same behavior, you have to create an
``add_arguments(cls, parser)`` class method on the test runner and call
``parser.add_argument`` to add any custom arguments, as parser is now an
:py:class:`argparse.ArgumentParser` instance.
Miscellaneous Miscellaneous
~~~~~~~~~~~~~ ~~~~~~~~~~~~~

View File

@ -333,9 +333,15 @@ execute and tear down the test suite.
runner, ensure it accepts ``**kwargs``. runner, ensure it accepts ``**kwargs``.
Your test runner may also define additional command-line options. Your test runner may also define additional command-line options.
If you add an ``option_list`` attribute to a subclassed test runner, Create or override an ``add_arguments(cls, parser)`` class method and add
those options will be added to the list of command-line options that custom arguments by calling ``parser.add_argument()`` inside the method, so
the :djadmin:`test` command can use. that the :djadmin:`test` command will be able to use those arguments.
.. versionchanged:: 1.8
Previously, you had to provide an ``option_list`` attribute to a
subclassed test runner to add options to the list of command-line
options that the :djadmin:`test` command could use.
Attributes Attributes
~~~~~~~~~~ ~~~~~~~~~~
@ -372,6 +378,12 @@ Attributes
management command's ``OptionParser`` for parsing arguments. See the management command's ``OptionParser`` for parsing arguments. See the
documentation for Python's ``optparse`` module for more details. documentation for Python's ``optparse`` module for more details.
.. deprecated:: 1.8
You should now override the :meth:`~DiscoverRunner.add_arguments` class
method to add custom arguments accepted by the :djadmin:`test`
management command.
Methods Methods
~~~~~~~ ~~~~~~~
@ -389,6 +401,15 @@ Methods
This method should return the number of tests that failed. This method should return the number of tests that failed.
.. classmethod:: DiscoverRunner.add_arguments(parser)
.. versionadded:: 1.8
Override this class method to add custom arguments accepted by the
:djadmin:`test` management command. See
:py:meth:`argparse.ArgumentParser.add_argument()` for details about adding
arguments to a parser.
.. method:: DiscoverRunner.setup_test_environment(**kwargs) .. method:: DiscoverRunner.setup_test_environment(**kwargs)
Sets up the test environment by calling Sets up the test environment by calling

View File

@ -3,7 +3,6 @@ Tests for django test runner
""" """
from __future__ import unicode_literals from __future__ import unicode_literals
from optparse import make_option
import unittest import unittest
from django.core.exceptions import ImproperlyConfigured from django.core.exceptions import ImproperlyConfigured
@ -152,11 +151,6 @@ class ManageCommandTests(unittest.TestCase):
class CustomOptionsTestRunner(runner.DiscoverRunner): class CustomOptionsTestRunner(runner.DiscoverRunner):
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): 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, super(CustomOptionsTestRunner, self).__init__(verbosity=verbosity, interactive=interactive,
@ -165,6 +159,12 @@ class CustomOptionsTestRunner(runner.DiscoverRunner):
self.option_b = option_b self.option_b = option_b
self.option_c = option_c self.option_c = option_c
@classmethod
def add_arguments(cls, parser):
parser.add_argument('--option_a', '-a', action='store', dest='option_a', default='1'),
parser.add_argument('--option_b', '-b', action='store', dest='option_b', default='2'),
parser.add_argument('--option_c', '-c', action='store', dest='option_c', default='3'),
def run_tests(self, test_labels, extra_tests=None, **kwargs): def run_tests(self, test_labels, extra_tests=None, **kwargs):
print("%s:%s:%s" % (self.option_a, self.option_b, self.option_c)) print("%s:%s:%s" % (self.option_a, self.option_b, self.option_c))