|
@@ -50,13 +50,15 @@ look like this:
|
|
|
from polls.models import Poll
|
|
|
|
|
|
class Command(BaseCommand):
|
|
|
- args = '<poll_id poll_id ...>'
|
|
|
help = 'Closes the specified poll for voting'
|
|
|
|
|
|
+ def add_arguments(self, parser):
|
|
|
+ parser.add_argument('poll_id', nargs='+', type=int)
|
|
|
+
|
|
|
def handle(self, *args, **options):
|
|
|
- for poll_id in args:
|
|
|
+ for poll_id in options['poll_id']:
|
|
|
try:
|
|
|
- poll = Poll.objects.get(pk=int(poll_id))
|
|
|
+ poll = Poll.objects.get(pk=poll_id)
|
|
|
except Poll.DoesNotExist:
|
|
|
raise CommandError('Poll "%s" does not exist' % poll_id)
|
|
|
|
|
@@ -65,6 +67,14 @@ look like this:
|
|
|
|
|
|
self.stdout.write('Successfully closed poll "%s"' % poll_id)
|
|
|
|
|
|
+Before Django 1.8, management commands were based on the :py:mod:`optparse`
|
|
|
+module, and positional arguments were passed in ``*args`` while optional
|
|
|
+arguments were passed in ``**options``. Now that management commands use
|
|
|
+:py:mod:`argparse` for argument parsing, all arguments are passed in
|
|
|
+``**options`` by default, unless you name your positional arguments to ``args``
|
|
|
+(compatibility mode). You are encouraged to exclusively use ``**options`` for
|
|
|
+new commands.
|
|
|
+
|
|
|
.. _management-commands-output:
|
|
|
|
|
|
.. note::
|
|
@@ -81,28 +91,34 @@ look like this:
|
|
|
The new custom command can be called using ``python manage.py closepoll
|
|
|
<poll_id>``.
|
|
|
|
|
|
-The ``handle()`` method takes zero or more ``poll_ids`` and sets ``poll.opened``
|
|
|
+The ``handle()`` method takes one or more ``poll_ids`` and sets ``poll.opened``
|
|
|
to ``False`` for each one. If the user referenced any nonexistent polls, a
|
|
|
:class:`CommandError` is raised. The ``poll.opened`` attribute does not exist
|
|
|
in the :doc:`tutorial</intro/tutorial01>` and was added to
|
|
|
``polls.models.Poll`` for this example.
|
|
|
|
|
|
+.. _custom-commands-options:
|
|
|
+
|
|
|
+Accepting optional arguments
|
|
|
+============================
|
|
|
+
|
|
|
The same ``closepoll`` could be easily modified to delete a given poll instead
|
|
|
-of closing it by accepting additional command line options. These custom options
|
|
|
-must be added to :attr:`~BaseCommand.option_list` like this:
|
|
|
+of closing it by accepting additional command line options. These custom
|
|
|
+options can be added in the :meth:`~BaseCommand.add_arguments` method like this:
|
|
|
|
|
|
.. code-block:: python
|
|
|
|
|
|
- from optparse import make_option
|
|
|
-
|
|
|
class Command(BaseCommand):
|
|
|
- option_list = BaseCommand.option_list + (
|
|
|
- make_option('--delete',
|
|
|
+ def add_arguments(self, parser):
|
|
|
+ # Positional arguments
|
|
|
+ parser.add_argument('poll_id', nargs='+', type=int)
|
|
|
+
|
|
|
+ # Named (optional) arguments
|
|
|
+ parser.add_argument('--delete',
|
|
|
action='store_true',
|
|
|
dest='delete',
|
|
|
default=False,
|
|
|
- help='Delete poll instead of closing it'),
|
|
|
- )
|
|
|
+ help='Delete poll instead of closing it')
|
|
|
|
|
|
def handle(self, *args, **options):
|
|
|
# ...
|
|
@@ -110,9 +126,15 @@ must be added to :attr:`~BaseCommand.option_list` like this:
|
|
|
poll.delete()
|
|
|
# ...
|
|
|
|
|
|
+.. versionchanged:: 1.8
|
|
|
+
|
|
|
+ Previously, only the standard :py:mod:`optparse` library was supported and
|
|
|
+ you would have to extend the command ``option_list`` variable with
|
|
|
+ ``optparse.make_option()``.
|
|
|
+
|
|
|
The option (``delete`` in our example) is available in the options dict
|
|
|
-parameter of the handle method. See the :py:mod:`optparse` Python documentation
|
|
|
-for more about ``make_option`` usage.
|
|
|
+parameter of the handle method. See the :py:mod:`argparse` Python documentation
|
|
|
+for more about ``add_argument`` usage.
|
|
|
|
|
|
In addition to being able to add custom command line options, all
|
|
|
:doc:`management commands</ref/django-admin>` can accept some
|
|
@@ -202,6 +224,12 @@ All attributes can be set in your derived class and can be used in
|
|
|
a list of application names might set this to '<app_label
|
|
|
app_label ...>'.
|
|
|
|
|
|
+ .. deprecated:: 1.8
|
|
|
+
|
|
|
+ This should be done now in the :meth:`~BaseCommand.add_arguments()`
|
|
|
+ method, by calling the ``parser.add_argument()`` method. See the
|
|
|
+ ``closepoll`` example above.
|
|
|
+
|
|
|
.. attribute:: BaseCommand.can_import_settings
|
|
|
|
|
|
A boolean indicating whether the command needs to be able to
|
|
@@ -215,11 +243,25 @@ All attributes can be set in your derived class and can be used in
|
|
|
help message when the user runs the command
|
|
|
``python manage.py help <command>``.
|
|
|
|
|
|
+.. attribute:: BaseCommand.missing_args_message
|
|
|
+
|
|
|
+.. versionadded:: 1.8
|
|
|
+
|
|
|
+ If your command defines mandatory positional arguments, you can customize
|
|
|
+ the message error returned in the case of missing arguments. The default is
|
|
|
+ output by :py:mod:`argparse` ("too few arguments").
|
|
|
+
|
|
|
.. attribute:: BaseCommand.option_list
|
|
|
|
|
|
This is the list of ``optparse`` options which will be fed
|
|
|
into the command's ``OptionParser`` for parsing arguments.
|
|
|
|
|
|
+ .. deprecated:: 1.8
|
|
|
+
|
|
|
+ You should now override the :meth:`~BaseCommand.add_arguments` method to
|
|
|
+ add custom arguments accepted by your command.
|
|
|
+ See :ref:`the example above <custom-commands-options>`.
|
|
|
+
|
|
|
.. attribute:: BaseCommand.output_transaction
|
|
|
|
|
|
A boolean indicating whether the command outputs SQL
|
|
@@ -287,6 +329,15 @@ the :meth:`~BaseCommand.handle` method must be implemented.
|
|
|
super(Command, self).__init__(*args, **kwargs)
|
|
|
# ...
|
|
|
|
|
|
+.. method:: BaseCommand.add_arguments(parser)
|
|
|
+
|
|
|
+.. versionadded:: 1.8
|
|
|
+
|
|
|
+ Entry point to add parser arguments to handle command line arguments passed
|
|
|
+ to the command. Custom commands should override this method to add both
|
|
|
+ positional and optional arguments accepted by the command. Calling
|
|
|
+ ``super()`` is not needed when directly subclassing ``BaseCommand``.
|
|
|
+
|
|
|
.. method:: BaseCommand.get_version()
|
|
|
|
|
|
Return the Django version, which should be correct for all
|