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:
parent
cbff097bd9
commit
4b4524291a
|
@ -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']
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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
|
||||||
~~~~~~~~~~~~~
|
~~~~~~~~~~~~~
|
||||||
|
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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))
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue