{
  "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 C is the most relevant because it directly involves a custom management command, similar to the CURRENT bug report. Both involve issues with command-line arguments and argument parsing, with Candidate C addressing a crash in the management command framework when subparsers are used. The structural similarity in handling command options and the potential for similar argument parsing issues make this candidate particularly helpful for debugging the CURRENT issue, as insights from Candidate C's patch could assist in resolving the customization problems present in the CURRENT bug.",
  "instance_id": "django__django-15781",
  "repo": "django/django",
  "created_at": "2022-06-18T19:39:34Z",
  "problem_statement": "Customizable management command formatters.\nDescription\n\t\nWith code like:\nclass Command(BaseCommand):\n\thelp = '''\n\tImport a contract from tzkt.\n\tExample usage:\n\t\t./manage.py tzkt_import 'Tezos Mainnet' KT1HTDtMBRCKoNHjfWEEvXneGQpCfPAt6BRe\n\t'''\nHelp output is:\n$ ./manage.py help tzkt_import\nusage: manage.py tzkt_import [-h] [--api API] [--version] [-v {0,1,2,3}] [--settings SETTINGS]\n\t\t\t\t\t\t\t [--pythonpath PYTHONPATH] [--traceback] [--no-color] [--force-color]\n\t\t\t\t\t\t\t [--skip-checks]\n\t\t\t\t\t\t\t blockchain target\nImport a contract from tzkt Example usage: ./manage.py tzkt_import 'Tezos Mainnet'\nKT1HTDtMBRCKoNHjfWEEvXneGQpCfPAt6BRe\npositional arguments:\n blockchain\t\t\tName of the blockchain to import into\n target\t\t\t\tId of the contract to import\nWhen that was expected:\n$ ./manage.py help tzkt_import\nusage: manage.py tzkt_import [-h] [--api API] [--version] [-v {0,1,2,3}] [--settings SETTINGS]\n\t\t\t\t\t\t\t [--pythonpath PYTHONPATH] [--traceback] [--no-color] [--force-color]\n\t\t\t\t\t\t\t [--skip-checks]\n\t\t\t\t\t\t\t blockchain target\nImport a contract from tzkt \nExample usage: \n\t./manage.py tzkt_import 'Tezos Mainnet' KT1HTDtMBRCKoNHjfWEEvXneGQpCfPAt6BRe\npositional arguments:\n blockchain\t\t\tName of the blockchain to import into\n target\t\t\t\tId of the contract to import\n",
  "patch": "diff --git a/django/core/management/base.py b/django/core/management/base.py\n--- a/django/core/management/base.py\n+++ b/django/core/management/base.py\n@@ -286,10 +286,10 @@ def create_parser(self, prog_name, subcommand, **kwargs):\n         Create and return the ``ArgumentParser`` which will be used to\n         parse the arguments to this command.\n         \"\"\"\n+        kwargs.setdefault(\"formatter_class\", DjangoHelpFormatter)\n         parser = CommandParser(\n             prog=\"%s %s\" % (os.path.basename(prog_name), subcommand),\n             description=self.help or None,\n-            formatter_class=DjangoHelpFormatter,\n             missing_args_message=getattr(self, \"missing_args_message\", None),\n             called_from_command_line=getattr(self, \"_called_from_command_line\", None),\n             **kwargs,\n"
}