{
  "instance_id": "django__django-14382",
  "repo": "django/django",
  "created_at": "2021-05-11T10:40:42Z",
  "problem_statement": "django-admin startapp with trailing slash in directory name results in error\nDescription\n\t\nBash tab-completion appends trailing slashes to directory names. django-admin startapp name directory/ results in the error:\nCommandError: '' is not a valid app directory. Please make sure the directory is a valid identifier.\nThe error is caused by ​line 77 of django/core/management/templates.py by calling basename() on the path with no consideration for a trailing slash:\nself.validate_name(os.path.basename(target), 'directory')\nRemoving potential trailing slashes would solve the problem:\nself.validate_name(os.path.basename(target.rstrip(os.sep)), 'directory')\n",
  "patch": "diff --git a/django/core/management/templates.py b/django/core/management/templates.py\n--- a/django/core/management/templates.py\n+++ b/django/core/management/templates.py\n@@ -73,9 +73,9 @@ def handle(self, app_or_project, name, target=None, **options):\n             except OSError as e:\n                 raise CommandError(e)\n         else:\n-            if app_or_project == 'app':\n-                self.validate_name(os.path.basename(target), 'directory')\n             top_dir = os.path.abspath(os.path.expanduser(target))\n+            if app_or_project == 'app':\n+                self.validate_name(os.path.basename(top_dir), 'directory')\n             if not os.path.exists(top_dir):\n                 raise CommandError(\"Destination directory '%s' does not \"\n                                    \"exist, please create it first.\" % top_dir)\n",
  "similar_bug_items": [
    {
      "pr_number": 11559,
      "pr_title": "Refs #30557 -- Fixed crash of ordering by ptr fields when Meta.ordering contains F() expressions.",
      "pr_body": "Thanks Can Sarıgöl for the report.\r\n\r\nFollow up to 8c5f9906c56ac72fc4f13218dd90bdf9bc8a248b",
      "issue_id": 30557,
      "issue_title": "order_by() a parent model crash when Meta.ordering contains expressions.",
      "issue_body": "Hi friends,\nDuring testing I discovered a strange bug when using a query expression for ordering during multi-table inheritance. You can find the full write up as well as reproducible test repository\n​\nhttps://github.com/JonnyWaffles/djangoordermetabug\n. The bug occurs because the field is an OrderBy object, not a string, during get_order_dir. The linked stacktrace should make the issue obvious, but what I don't understand is why it only fails during test db setup, not during repl or script use. I wish I could help more and come up with a real solution. Hopefully, this is enough for someone wiser to find the culprit.",
      "issue_closed_at": "2019-07-11T05:07:44",
      "base_commit": "8c5f9906c56ac72fc4f13218dd90bdf9bc8a248b",
      "changes": [
        {
          "file": "django/db/models/sql/compiler.py",
          "type": "function",
          "name": "find_ordering_name",
          "class_name": "SQLCompiler",
          "code": "def find_ordering_name(self, name, opts, alias=None, default_order='ASC',\n                           already_seen=None):\n        \"\"\"\n        Return the table alias (the name might be ambiguous, the alias will\n        not be) and column name for ordering by the given 'name' parameter.\n        The 'name' is of the form 'field1__field2__...__fieldN'.\n        \"\"\"\n        name, order = get_order_dir(name, default_order)\n        descending = order == 'DESC'\n        pieces = name.split(LOOKUP_SEP)\n        field, targets, alias, joins, path, opts, transform_function = self._setup_joins(pieces, opts, alias)\n\n        # If we get to this point and the field is a relation to another model,\n        # append the default ordering for that model unless the attribute name\n        # of the field is specified.\n        if field.is_relation and opts.ordering and getattr(field, 'attname', None) != name:\n            # Firstly, avoid infinite loops.\n            already_seen = already_seen or set()\n            join_tuple = tuple(getattr(self.query.alias_map[j], 'join_cols', None) for j in joins)\n            if join_tuple in already_seen:\n                raise FieldError('Infinite loop caused by ordering.')\n            already_seen.add(join_tuple)\n\n            results = []\n            for item in opts.ordering:\n                if isinstance(item, OrderBy):\n                    results.append((item, False))\n                    continue\n                results.extend(self.find_ordering_name(item, opts, alias,\n                                                       order, already_seen))\n            return results\n        targets, alias, _ = self.query.trim_joins(targets, joins, path)\n        return [(OrderBy(transform_function(t, alias), descending=descending), False) for t in targets]"
        }
      ]
    },
    {
      "pr_number": 12332,
      "pr_title": "Fixed #30439 - Added support for different plurals for the same language.",
      "pr_body": "ticket-30439",
      "issue_id": 30439,
      "issue_title": "Translations issues on Django upgrade due to unexpected changes in plural forms",
      "issue_body": "When locales have different plural forms, the ngettext can be easily broken because it uses plural equation from the first loaded gettext catalog for all of them.\nFor example in Czech locale, there are different plural equations being used even inside Django with either 3 or 4 plural forms. When the one with 4 forms is loaded first, Django is looking for non existing plural in catalogs.\nReproducer in Django 2.2.1:\n>>> from django.utils import translation\n>>> translation.activate('cs')\n>>> translation.ngettext('This password is too short. It must contain at least %(min_length)d character.', 'This password is too short. It must contain at least %(min_length)d characters.', 1)\n'Heslo je příliš krátké. Musí mít délku aspoň %(min_length)d znak.'\n>>> translation.ngettext('This password is too short. It must contain at least %(min_length)d character.', 'This password is too short. It must contain at least %(min_length)d characters.', 10)\n'This password is too short. It must contain at least %(min_length)d characters.'\nThe second invocation is trying to find 4th plural in 3 plurals catalog.",
      "issue_closed_at": "2020-03-10T09:56:53",
      "base_commit": "591e2270dc8c685625be25dbed908a9a3897ba1d",
      "changes": [
        {
          "file": "django/utils/translation/trans_real.py",
          "type": "function",
          "name": "reset_cache",
          "class_name": null,
          "code": "def reset_cache(**kwargs):\n    \"\"\"\n    Reset global state when LANGUAGES setting has been changed, as some\n    languages should no longer be accepted.\n    \"\"\"\n    if kwargs['setting'] in ('LANGUAGES', 'LANGUAGE_CODE'):\n        check_for_language.cache_clear()\n        get_languages.cache_clear()\n        get_supported_language_variant.cache_clear()"
        },
        {
          "file": "django/utils/translation/trans_real.py",
          "type": "function",
          "name": "__init__",
          "class_name": "DjangoTranslation",
          "code": "def __init__(self, language, domain=None, localedirs=None):\n        \"\"\"Create a GNUTranslations() using many locale directories\"\"\"\n        gettext_module.GNUTranslations.__init__(self)\n        if domain is not None:\n            self.domain = domain\n\n        self.__language = language\n        self.__to_language = to_language(language)\n        self.__locale = to_locale(language)\n        self._catalog = None\n        # If a language doesn't have a catalog, use the Germanic default for\n        # pluralization: anything except one is pluralized.\n        self.plural = lambda n: int(n != 1)\n\n        if self.domain == 'django':\n            if localedirs is not None:\n                # A module-level cache is used for caching 'django' translations\n                warnings.warn(\"localedirs is ignored when domain is 'django'.\", RuntimeWarning)\n                localedirs = None\n            self._init_translation_catalog()\n\n        if localedirs:\n            for localedir in localedirs:\n                translation = self._new_gnu_trans(localedir)\n                self.merge(translation)\n        else:\n            self._add_installed_apps_translations()\n\n        self._add_local_translations()\n        if self.__language == settings.LANGUAGE_CODE and self.domain == 'django' and self._catalog is None:\n            # default lang should have at least one translation file available.\n            raise OSError('No translation files found for default language %s.' % settings.LANGUAGE_CODE)\n        self._add_fallback(localedirs)\n        if self._catalog is None:\n            # No catalogs found for this language, set an empty catalog.\n            self._catalog = {}"
        },
        {
          "file": "django/utils/translation/trans_real.py",
          "type": "function",
          "name": "merge",
          "class_name": "DjangoTranslation",
          "code": "def merge(self, other):\n        \"\"\"Merge another translation into this catalog.\"\"\"\n        if not getattr(other, '_catalog', None):\n            return  # NullTranslations() has no _catalog\n        if self._catalog is None:\n            # Take plural and _info from first catalog found (generally Django's).\n            self.plural = other.plural\n            self._info = other._info.copy()\n            self._catalog = other._catalog.copy()\n        else:\n            self._catalog.update(other._catalog)\n        if other._fallback:\n            self.add_fallback(other._fallback)"
        },
        {
          "file": "django/utils/translation/trans_real.py",
          "type": "function",
          "name": "to_language",
          "class_name": "DjangoTranslation",
          "code": "def to_language(self):\n        \"\"\"Return the translation language name.\"\"\"\n        return self.__to_language"
        }
      ]
    },
    {
      "pr_number": 12263,
      "pr_title": "Fixed #31166 -- Used \"raise from\" when raising ImproperlyConfigured exceptions in django.urls.resolvers.",
      "pr_body": "This will make it have a text of \"this exception is the direct result of\" instead of \"During handling of the above exception, another exception occurred\" \r\n\r\nThis is more accurate for the case of this exception.",
      "issue_id": 31166,
      "issue_title": "Provide context for ImproperlyConfigured exceptions in URL resolver.",
      "issue_body": "The patch is ready here:\n​\nhttps://github.com/django/django/pull/12263\nThis will make it have a text of \"this exception is the direct result of\" instead of \"During handling of the above exception, another exception occurred\"\nThis is more accurate for the case of this exception.\nIf this change will be merged, there are a couple more place in this module where it can be added.\nWhen it happened to me a few weeks ago, it was due to a circular import, which I then fixed. Now of course, when I tried to reproduce the circular import to show you, I couldn't. But I could achieve the same thing by renaming\nurlpatterns\nto\nxurlpatterns\n, so it wouldn't be available. This is the error I get:\nException in thread django-main-thread:\nTraceback (most recent call last):\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\urls\\resolvers.py\", line 590, in url_patterns\n    iter(patterns)\nTypeError: 'module' object is not iterable\n\nDuring handling of the above exception, another exception occurred:\n\nTraceback (most recent call last):\n  File \"C:\\Program Files\\Python38\\lib\\threading.py\", line 932, in _bootstrap_inner\n    self.run()\n  File \"C:\\Program Files\\Python38\\lib\\threading.py\", line 870, in run\n    self._target(*self._args, **self._kwargs)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\utils\\autoreload.py\", line 53, in wrapper\n    fn(*args, **kwargs)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\management\\commands\\runserver.py\", line 117, in inner_run\n    self.check(display_num_errors=True)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\management\\base.py\", line 392, in check\n    all_issues = self._run_checks(\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\management\\base.py\", line 382, in _run_checks\n    return checks.run_checks(**kwargs)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\checks\\registry.py\", line 72, in run_checks\n    new_errors = check(app_configs=app_configs)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\checks\\urls.py\", line 13, in check_url_config\n    return check_resolver(resolver)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\checks\\urls.py\", line 23, in check_resolver\n    return check_method()\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\urls\\resolvers.py\", line 408, in check\n    messages.extend(check_resolver(pattern))\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\checks\\urls.py\", line 23, in check_resolver\n    return check_method()\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\urls\\resolvers.py\", line 407, in check\n    for pattern in self.url_patterns:\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\utils\\functional.py\", line 48, in __get__\n    res = instance.__dict__[self.name] = self.func(instance)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\urls\\resolvers.py\", line 597, in url_patterns\n    raise ImproperlyConfigured(msg.format(name=self.urlconf_name))\ndjango.core.exceptions.ImproperlyConfigured: The included URLconf '<module 'barb.urls' from 'C:\\\\Users\\\\Administrator\\\\Desktop\\\\foof\\\\barb\\\\urls.py'>' does not appear to have any patterns in it. If you see valid patterns in the file then the issue is probably caused by a circular import.\nThe bad part is\nDuring handling of the above exception, another exception occurred\n.\nAfter my fix, it looks like this:\nException in thread django-main-thread:\nTraceback (most recent call last):\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\urls\\resolvers.py\", line 590, in url_patterns\n    iter(patterns)\nTypeError: 'module' object is not iterable\n\nThe above exception was the direct cause of the following exception:\n\nTraceback (most recent call last):\n  File \"C:\\Program Files\\Python38\\lib\\threading.py\", line 932, in _bootstrap_inner\n    self.run()\n  File \"C:\\Program Files\\Python38\\lib\\threading.py\", line 870, in run\n    self._target(*self._args, **self._kwargs)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\utils\\autoreload.py\", line 53, in wrapper\n    fn(*args, **kwargs)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\management\\commands\\runserver.py\", line 117, in inner_run\n    self.check(display_num_errors=True)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\management\\base.py\", line 392, in check\n    all_issues = self._run_checks(\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\management\\base.py\", line 382, in _run_checks\n    return checks.run_checks(**kwargs)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\checks\\registry.py\", line 72, in run_checks\n    new_errors = check(app_configs=app_configs)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\checks\\urls.py\", line 13, in check_url_config\n    return check_resolver(resolver)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\checks\\urls.py\", line 23, in check_resolver\n    return check_method()\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\urls\\resolvers.py\", line 408, in check\n    messages.extend(check_resolver(pattern))\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\checks\\urls.py\", line 23, in check_resolver\n    return check_method()\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\urls\\resolvers.py\", line 407, in check\n    for pattern in self.url_patterns:\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\utils\\functional.py\", line 48, in __get__\n    res = instance.__dict__[self.name] = self.func(instance)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\urls\\resolvers.py\", line 597, in url_patterns\n    raise ImproperlyConfigured(msg.format(name=self.urlconf_name)) from type_error # Ram hack, remove\ndjango.core.exceptions.ImproperlyConfigured: The included URLconf '<module 'barb.urls' from 'C:\\\\Users\\\\Administrator\\\\Desktop\\\\foof\\\\barb\\\\urls.py'>' does not appear to have any patterns in it. If you see valid patterns in the file then the issue is probably caused by a circular import.\nNow the text is corrected to\nThe above exception was the direct cause of the following exception",
      "issue_closed_at": "2020-01-17T05:21:04",
      "base_commit": "73563183c2ea92e9ef1d3a1f790a503acc14ade2",
      "changes": [
        {
          "file": "django/urls/resolvers.py",
          "type": "function",
          "name": "_compile",
          "class_name": "RoutePattern",
          "code": "def _compile(self, route):\n        return re.compile(_route_to_regex(route, self._is_endpoint)[0])"
        },
        {
          "file": "django/urls/resolvers.py",
          "type": "function",
          "name": "_route_to_regex",
          "class_name": null,
          "code": "def _route_to_regex(route, is_endpoint=False):\n    \"\"\"\n    Convert a path pattern into a regular expression. Return the regular\n    expression and a dictionary mapping the capture names to the converters.\n    For example, 'foo/<int:pk>' returns '^foo\\\\/(?P<pk>[0-9]+)'\n    and {'pk': <django.urls.converters.IntConverter>}.\n    \"\"\"\n    if not set(route).isdisjoint(string.whitespace):\n        raise ImproperlyConfigured(\"URL route '%s' cannot contain whitespace.\" % route)\n    original_route = route\n    parts = ['^']\n    converters = {}\n    while True:\n        match = _PATH_PARAMETER_COMPONENT_RE.search(route)\n        if not match:\n            parts.append(re.escape(route))\n            break\n        parts.append(re.escape(route[:match.start()]))\n        route = route[match.end():]\n        parameter = match.group('parameter')\n        if not parameter.isidentifier():\n            raise ImproperlyConfigured(\n                \"URL route '%s' uses parameter name %r which isn't a valid \"\n                \"Python identifier.\" % (original_route, parameter)\n            )\n        raw_converter = match.group('converter')\n        if raw_converter is None:\n            # If a converter isn't specified, the default is `str`.\n            raw_converter = 'str'\n        try:\n            converter = get_converter(raw_converter)\n        except KeyError as e:\n            raise ImproperlyConfigured(\n                \"URL route '%s' uses invalid converter %s.\" % (original_route, e)\n            )\n        converters[parameter] = converter\n        parts.append('(?P<' + parameter + '>' + converter.regex + ')')\n    if is_endpoint:\n        parts.append('$')\n    return ''.join(parts), converters"
        },
        {
          "file": "django/urls/resolvers.py",
          "type": "function",
          "name": "url_patterns",
          "class_name": "URLResolver",
          "code": "def url_patterns(self):\n        # urlconf_module might be a valid set of patterns, so we default to it\n        patterns = getattr(self.urlconf_module, \"urlpatterns\", self.urlconf_module)\n        try:\n            iter(patterns)\n        except TypeError:\n            msg = (\n                \"The included URLconf '{name}' does not appear to have any \"\n                \"patterns in it. If you see valid patterns in the file then \"\n                \"the issue is probably caused by a circular import.\"\n            )\n            raise ImproperlyConfigured(msg.format(name=self.urlconf_name))\n        return patterns"
        }
      ]
    },
    {
      "pr_number": 7471,
      "pr_title": "Fixed #26812 -- Check only path with APPEND_SLASH enabled",
      "pr_body": "",
      "issue_id": 26812,
      "issue_title": "APPEND_SLASH doesn't work with URLs that have query strings",
      "issue_body": "Hi,\nIn\nmiddleware/common.py\nthere is a code, which checks whether the slash should be appended to URL.\ndef should_redirect_with_slash(self, request):\n        \"\"\"\n        Return True if settings.APPEND_SLASH is True and appending a slash to\n        the request path turns an invalid path into a valid one.\n        \"\"\"\n\n        if settings.APPEND_SLASH and not request.get_full_path().endswith('/'):\n            urlconf = getattr(request, 'urlconf', None)\n\n            return (\n                not urlresolvers.is_valid_path(request.path_info, urlconf)\n                and urlresolvers.is_valid_path('%s/' % request.path_info, urlconf)\n            )\n        return False\nIn most cases it works fine, but it fails to append slash, when used with LOGIN_URL without slash and with \"next\" parameter. For example:\nIt works fine:\n/accounts/login/?next=/polls/3/\n-> LOGIN_URL =\n/accounts/login/\nBut this one returns 404:\n/accounts/login?next=/polls/3/\n-> LOGIN_URL =\n/accounts/login (no trailing slash).\nIt is because:\nrequest.get_full_path().endswith('/')\nchecks also query string and therefore returns true and the slash is never appended.",
      "issue_closed_at": "2016-11-06T03:38:42",
      "base_commit": "b741fe397aa567dd43d3e9dbd9fb5ecca6eab412",
      "changes": [
        {
          "file": "django/middleware/common.py",
          "type": "function",
          "name": "should_redirect_with_slash",
          "class_name": "CommonMiddleware",
          "code": "def should_redirect_with_slash(self, request):\n        \"\"\"\n        Return True if settings.APPEND_SLASH is True and appending a slash to\n        the request path turns an invalid path into a valid one.\n        \"\"\"\n        if settings.APPEND_SLASH and not request.get_full_path().endswith('/'):\n            urlconf = getattr(request, 'urlconf', None)\n            return (\n                not is_valid_path(request.path_info, urlconf) and\n                is_valid_path('%s/' % request.path_info, urlconf)\n            )\n        return False"
        }
      ]
    },
    {
      "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"
        }
      ]
    }
  ]
}