{
  "Selected_candidate": {
    "pr_number": 9894,
    "pr_title": "Fixed #29295 -- Fixed management command crash when using subparsers.",
    "pr_body": "I applied  @timgraham  [diff](https://code.djangoproject.com/attachment/ticket/29295/29295.diff), and added tests for it.",
    "issue_id": 29295,
    "issue_title": "BaseCommand.add_arguments crashes when using parser.add_subparsers().add_parser(\"subcommand\")",
    "issue_body": "Background\nPython's\n​\nargparse\nallows to\n​\ndefine subparsers\n.\nThe following works (but does nothing observable) in plain Python:\nimport argparse\n\nparser = argparse.ArgumentParser()\nsubparsers = parser.add_subparsers()\na_parser = subparsers.add_parser(\"A\")\nSetup\nThe equivalent does\nnot\nwork in a\nmanage.py\ncustom command.\nWhen I put this in file\nmycommand.py\n:\nimport django.core.management.base as djcmb\n\nclass Command(djcmb.BaseCommand):\n    def add_arguments(self, parser):\n        subparsers = parser.add_subparsers()\n        a_parser = subparsers.add_parser(\"A\")\nSymptom\n...and then run\npython manage.py mycommand A\n,\nwhat I get is\nTypeError: __init__() missing 1 required positional argument: 'cmd'\nwith the following stacktrace:\nFile \"manage.py\", line 8, in <module>\n    execute_from_command_line(sys.argv)\n  File \"C:\\venv\\rqc36\\lib\\site-packages\\django\\core\\management\\__init__.py\", line 364, in execute_from_command_line\n    utility.execute()\n  File \"C:\\venv\\rqc36\\lib\\site-packages\\django\\core\\management\\__init__.py\", line 356, in execute\n    self.fetch_command(subcommand).run_from_argv(self.argv)\n  File \"C:\\venv\\rqc36\\lib\\site-packages\\django\\core\\management\\base.py\", line 275, in run_from_argv\n    parser = self.create_parser(argv[0], argv[1])\n  File \"C:\\venv\\rqc36\\lib\\site-packages\\django\\core\\management\\base.py\", line 249, in create_parser\n    self.add_arguments(parser)\n  File \"C:\\ws\\bb\\rqc\\rqc\\management\\commands\\mycommand.py\", line 7, in add_arguments\n    a_parser = subparsers.add_parser(\"A\")\n  File \"C:\\sw\\Python36-32\\lib\\argparse.py\", line 1097, in add_parser\n    parser = self._parser_class(**kwargs)\nDiagnosis\nThe reason is that Django does not use a plain\nArgumentParser\nbut rather its own\ndjango.core.management.base.CommandParser\n, the constructor of which requires a positional argument:\ndef __init__(self, cmd, **kwargs):  # ...\nwhere\ncmd\nis supposed to contain the\nmycommand.Command\nobject which is then stored as\nself.cmd\nin the parser.\nThe following\nkludge\nhelps the poor user trying to get subparsers to work:\nimport argparse\nimport django.core.management.base as djcmb\n\nclass Command(djcmb.BaseCommand):\n    def add_arguments(self, parser):\n        subparsers = parser.add_subparsers()\n        subparsers._parser_class = argparse.ArgumentParser  # circumvent Django 1.11 bug\n        a_parser = subparsers.add_parser(\"A\")\nRepair suggestions\nS1\n: The most straightforward repair I see would be to make the argument optional (\n*argv\n) and use a dummy object for\nself.cmd\nin case of subparsers.\nself.cmd\nis used only three times. Is this good enough?\nS2\n: Alternatively, one could simply document the problem and the kludge. (BTW: The custom commands how-to should become more explicit regarding the use of\nargparse\n; it is currently only hinted at, never mentioned.)",
    "issue_closed_at": "2018-04-21T16:59:28",
    "base_commit": "21420096c4db78ccb8f549a29d662cff870d363c",
    "changes": [
      {
        "file": "django/core/management/__init__.py",
        "type": "function",
        "name": "execute",
        "class_name": "ManagementUtility",
        "code": "def execute(self):\n        \"\"\"\n        Given the command-line arguments, figure out which subcommand is being\n        run, create a parser appropriate to that command, and run it.\n        \"\"\"\n        try:\n            subcommand = self.argv[1]\n        except IndexError:\n            subcommand = 'help'  # Display help if no arguments were given.\n\n        # Preprocess options to extract --settings and --pythonpath.\n        # These options could affect the commands that are available, so they\n        # must be processed early.\n        parser = CommandParser(None, usage=\"%(prog)s subcommand [options] [args]\", add_help=False)\n        parser.add_argument('--settings')\n        parser.add_argument('--pythonpath')\n        parser.add_argument('args', nargs='*')  # catch-all\n        try:\n            options, args = parser.parse_known_args(self.argv[2:])\n            handle_default_options(options)\n        except CommandError:\n            pass  # Ignore any option errors at this point.\n\n        try:\n            settings.INSTALLED_APPS\n        except ImproperlyConfigured as exc:\n            self.settings_exception = exc\n        except ImportError as exc:\n            self.settings_exception = exc\n\n        if settings.configured:\n            # Start the auto-reloading dev server even if the code is broken.\n            # The hardcoded condition is a code smell but we can't rely on a\n            # flag on the command class because we haven't located it yet.\n            if subcommand == 'runserver' and '--noreload' not in self.argv:\n                try:\n                    autoreload.check_errors(django.setup)()\n                except Exception:\n                    # The exception will be raised later in the child process\n                    # started by the autoreloader. Pretend it didn't happen by\n                    # loading an empty list of applications.\n                    apps.all_models = defaultdict(OrderedDict)\n                    apps.app_configs = OrderedDict()\n                    apps.apps_ready = apps.models_ready = apps.ready = True\n\n                    # Remove options not compatible with the built-in runserver\n                    # (e.g. options for the contrib.staticfiles' runserver).\n                    # Changes here require manually testing as described in\n                    # #27522.\n                    _parser = self.fetch_command('runserver').create_parser('django', 'runserver')\n                    _options, _args = _parser.parse_known_args(self.argv[2:])\n                    for _arg in _args:\n                        self.argv.remove(_arg)\n\n            # In all other cases, django.setup() is required to succeed.\n            else:\n                django.setup()\n\n        self.autocomplete()\n\n        if subcommand == 'help':\n            if '--commands' in args:\n                sys.stdout.write(self.main_help_text(commands_only=True) + '\\n')\n            elif not options.args:\n                sys.stdout.write(self.main_help_text() + '\\n')\n            else:\n                self.fetch_command(options.args[0]).print_help(self.prog_name, options.args[0])\n        # Special-cases: We want 'django-admin --version' and\n        # 'django-admin --help' to work, for backwards compatibility.\n        elif subcommand == 'version' or self.argv[1:] == ['--version']:\n            sys.stdout.write(django.get_version() + '\\n')\n        elif self.argv[1:] in (['--help'], ['-h']):\n            sys.stdout.write(self.main_help_text() + '\\n')\n        else:\n            self.fetch_command(subcommand).run_from_argv(self.argv)"
      },
      {
        "file": "django/core/management/base.py",
        "type": "class",
        "name": "CommandParser",
        "code": "class CommandParser(ArgumentParser):\n    \"\"\"\n    Customized ArgumentParser class to improve some error messages and prevent\n    SystemExit in several occasions, as SystemExit is unacceptable when a\n    command is called programmatically.\n    \"\"\"\n    def __init__(self, cmd, **kwargs):\n        self.cmd = cmd\n        super().__init__(**kwargs)\n\n    def parse_args(self, args=None, namespace=None):\n        # Catch missing argument for a better error message\n        if (hasattr(self.cmd, 'missing_args_message') and\n                not (args or any(not arg.startswith('-') for arg in args))):\n            self.error(self.cmd.missing_args_message)\n        return super().parse_args(args, namespace)\n\n    def error(self, message):\n        if self.cmd._called_from_command_line:\n            super().error(message)\n        else:\n            raise CommandError(\"Error: %s\" % message)"
      },
      {
        "file": "django/core/management/base.py",
        "type": "function",
        "name": "create_parser",
        "class_name": "BaseCommand",
        "code": "def create_parser(self, prog_name, subcommand):\n        \"\"\"\n        Create and return the ``ArgumentParser`` which will be used to\n        parse the arguments to this command.\n        \"\"\"\n        parser = CommandParser(\n            self, prog=\"%s %s\" % (os.path.basename(prog_name), subcommand),\n            description=self.help or None,\n        )\n        # Add command-specific arguments first so that they appear in the\n        # --help output before arguments common to all commands.\n        self.add_arguments(parser)\n        parser.add_argument('--version', action='version', version=self.get_version())\n        parser.add_argument(\n            '-v', '--verbosity', action='store', dest='verbosity', default=1,\n            type=int, choices=[0, 1, 2, 3],\n            help='Verbosity level; 0=minimal output, 1=normal output, 2=verbose output, 3=very verbose output',\n        )\n        parser.add_argument(\n            '--settings',\n            help=(\n                'The Python path to a settings module, e.g. '\n                '\"myproject.settings.main\". If this isn\\'t provided, the '\n                'DJANGO_SETTINGS_MODULE environment variable will be used.'\n            ),\n        )\n        parser.add_argument(\n            '--pythonpath',\n            help='A directory to add to the Python path, e.g. \"/home/djangoprojects/myproject\".',\n        )\n        parser.add_argument('--traceback', action='store_true', help='Raise on CommandError exceptions')\n        parser.add_argument(\n            '--no-color', action='store_true', dest='no_color',\n            help=\"Don't colorize the command output.\",\n        )\n        return parser"
      }
    ]
  },
  "Justification": "Candidate A is the most helpful because it involves similar components of the Django management utility. Both the CURRENT bug and Candidate A's bug report involve the `CommandParser` and its interaction with argument parsing in a Django CLI context. The structural similarity in the stack traces and the symptoms of errors stemming from incorrect handling of command-line arguments provide significant insight into the CURRENT bug's issue of improper argument usage. The discussions around fixing the `add_arguments` method could inform how to appropriately modify the `CommandParser` in the CURRENT bug.",
  "instance_id": "django__django-13658",
  "repo": "django/django",
  "created_at": "2020-11-09T20:50:28Z",
  "problem_statement": "ManagementUtility instantiates CommandParser without passing already-computed prog argument\nDescription\n\t\nManagementUtility ​goes to the trouble to parse the program name from the argv it's passed rather than from sys.argv: \n\tdef __init__(self, argv=None):\n\t\tself.argv = argv or sys.argv[:]\n\t\tself.prog_name = os.path.basename(self.argv[0])\n\t\tif self.prog_name == '__main__.py':\n\t\t\tself.prog_name = 'python -m django'\nBut then when it needs to parse --pythonpath and --settings, it ​uses the program name from sys.argv: \n\t\tparser = CommandParser(usage='%(prog)s subcommand [options] [args]', add_help=False, allow_abbrev=False)\nAbove \"%(prog)s\" ​refers to sys.argv[0]. Instead, it should refer to self.prog_name. This can fixed as follows:\n\t\tparser = CommandParser(\n\t\t\tprog=self.prog_name,\n\t\t\tusage='%(prog)s subcommand [options] [args]',\n\t\t\tadd_help=False,\n\t\t\tallow_abbrev=False)\nI'm aware that execute_from_command_line is a private API, but it'd be really convenient for me if it worked properly in my weird embedded environment where sys.argv[0] is ​incorrectly None. If passing my own argv to execute_from_command_line avoided all the ensuing exceptions, I wouldn't have to modify sys.argv[0] globally as I'm doing in the meantime.\n",
  "patch": "diff --git a/django/core/management/__init__.py b/django/core/management/__init__.py\n--- a/django/core/management/__init__.py\n+++ b/django/core/management/__init__.py\n@@ -344,7 +344,12 @@ def execute(self):\n         # Preprocess options to extract --settings and --pythonpath.\n         # These options could affect the commands that are available, so they\n         # must be processed early.\n-        parser = CommandParser(usage='%(prog)s subcommand [options] [args]', add_help=False, allow_abbrev=False)\n+        parser = CommandParser(\n+            prog=self.prog_name,\n+            usage='%(prog)s subcommand [options] [args]',\n+            add_help=False,\n+            allow_abbrev=False,\n+        )\n         parser.add_argument('--settings')\n         parser.add_argument('--pythonpath')\n         parser.add_argument('args', nargs='*')  # catch-all\n"
}