{
  "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",
  "similar_bug_items": [
    {
      "pr_number": 6754,
      "pr_title": "Fixed #26736 -- Improved unicode handling for SpatialReference.",
      "pr_body": "https://code.djangoproject.com/ticket/26736\n",
      "issue_id": 26736,
      "issue_title": "SpatialReference crashes when initialized with WKT containining unicode characters",
      "issue_body": "from\ndjango.contrib.gis.db.backends.spatialite.models\nimport\nSpatialiteSpatialRefSys\nas\nSpatialRefSys\nfrom\ndjango.contrib.gis.gdal.srs\nimport\nSpatialReference\nSpatialReference\n(\nSpatialRefSys\n.\nobjects\n.\nget\n(\nsrid\n=\n187939\n)\n.\nsrtext\n)\n/home/sergey/dev/django/django/contrib/gis/gdal/srs.py in __init__(self, srs_input, srs_type)\n     58             # Encoding to ASCII if unicode passed in.\n     59             if isinstance(srs_input, six.text_type):\n---> 60                 srs_input = srs_input.encode('ascii')\n     61             try:\n     62                 # If SRID is a string, e.g., '4326', then make acceptable\n\nUnicodeEncodeError: 'ascii' codec can't encode character u'\\xdf' in position 34: ordinal not in range(128)",
      "issue_closed_at": "2016-06-11T20:00:40",
      "base_commit": "0451dcc2eb2449a988ade8e603846f0508ce76b4",
      "changes": [
        {
          "file": "django/contrib/gis/gdal/prototypes/srs.py",
          "type": "function",
          "name": "units_func",
          "class_name": null,
          "code": "def units_func(f):\n    \"\"\"\n    Creates a ctypes function prototype for OSR units functions, e.g.,\n    OSRGetAngularUnits, OSRGetLinearUnits.\n    \"\"\"\n    return double_output(f, [c_void_p, POINTER(c_char_p)], strarg=True)"
        },
        {
          "file": "django/contrib/gis/gdal/srs.py",
          "type": "function",
          "name": "__init__",
          "class_name": "CoordTransform",
          "code": "def __init__(self, source, target):\n        \"Initializes on a source and target SpatialReference objects.\"\n        if not isinstance(source, SpatialReference) or not isinstance(target, SpatialReference):\n            raise TypeError('source and target must be of type SpatialReference')\n        self.ptr = capi.new_ct(source._ptr, target._ptr)\n        self._srs1_name = source.name\n        self._srs2_name = target.name"
        },
        {
          "file": "django/contrib/gis/gdal/srs.py",
          "type": "function",
          "name": "import_user_input",
          "class_name": "SpatialReference",
          "code": "def import_user_input(self, user_input):\n        \"Imports the Spatial Reference from the given user input string.\"\n        capi.from_user_input(self.ptr, force_bytes(user_input))"
        },
        {
          "file": "django/contrib/gis/gdal/srs.py",
          "type": "function",
          "name": "proj4",
          "class_name": "SpatialReference",
          "code": "def proj4(self):\n        \"Alias for proj().\"\n        return self.proj"
        }
      ]
    },
    {
      "pr_number": 8329,
      "pr_title": "Fixed #28052 -- Prevented dropping Meta.indexes when changing db_index to False.",
      "pr_body": "https://code.djangoproject.com/ticket/28052",
      "issue_id": 28052,
      "issue_title": "Django inadvertently removes all indexes when switching to db_index=False",
      "issue_body": "Create a model with\ndata = JSONField(db_index=True)\n; makemigrations; migrate\nAdd a\nGinIndex(fields=['data'], name='foo')\nto\n_meta.indexes\n; makemigrations; migrate\nRemove\ndb_index=True\nfrom the\nJSONField\n; makemigrations; migrate\nAfter step 2 there will be a Gin index on the\ndata\ncolumn on the model. After step 3 there's _no_ index on the\ndata\ncolumn anymore.",
      "issue_closed_at": "2017-05-01T11:21:23",
      "base_commit": "63afe3a2bfaf97fecff6641137a85296029d5b73",
      "changes": [
        {
          "file": "django/db/backends/base/schema.py",
          "type": "line",
          "name": "line 3",
          "code": "from datetime import datetime\n\nfrom django.db.backends.utils import strip_quotes\nfrom django.db.transaction import TransactionManagementError, atomic\nfrom django.utils import timezone\nfrom django.utils.encoding import force_bytes"
        },
        {
          "file": "django/db/backends/base/schema.py",
          "type": "function",
          "name": "_alter_field",
          "class_name": "BaseDatabaseSchemaEditor",
          "code": "def _alter_field(self, model, old_field, new_field, old_type, new_type,\n                     old_db_params, new_db_params, strict=False):\n        \"\"\"Perform a \"physical\" (non-ManyToMany) field update.\"\"\"\n        # Drop any FK constraints, we'll remake them later\n        fks_dropped = set()\n        if old_field.remote_field and old_field.db_constraint:\n            fk_names = self._constraint_names(model, [old_field.column], foreign_key=True)\n            if strict and len(fk_names) != 1:\n                raise ValueError(\"Found wrong number (%s) of foreign key constraints for %s.%s\" % (\n                    len(fk_names),\n                    model._meta.db_table,\n                    old_field.column,\n                ))\n            for fk_name in fk_names:\n                fks_dropped.add((old_field.column,))\n                self.execute(self._delete_constraint_sql(self.sql_delete_fk, model, fk_name))\n        # Has unique been removed?\n        if old_field.unique and (not new_field.unique or (not old_field.primary_key and new_field.primary_key)):\n            # Find the unique constraint for this field\n            constraint_names = self._constraint_names(model, [old_field.column], unique=True)\n            if strict and len(constraint_names) != 1:\n                raise ValueError(\"Found wrong number (%s) of unique constraints for %s.%s\" % (\n                    len(constraint_names),\n                    model._meta.db_table,\n                    old_field.column,\n                ))\n            for constraint_name in constraint_names:\n                self.execute(self._delete_constraint_sql(self.sql_delete_unique, model, constraint_name))\n        # Drop incoming FK constraints if we're a primary key and things are going\n        # to change.\n        if old_field.primary_key and new_field.primary_key and old_type != new_type:\n            # '_meta.related_field' also contains M2M reverse fields, these\n            # will be filtered out\n            for _old_rel, new_rel in _related_non_m2m_objects(old_field, new_field):\n                rel_fk_names = self._constraint_names(\n                    new_rel.related_model, [new_rel.field.column], foreign_key=True\n                )\n                for fk_name in rel_fk_names:\n                    self.execute(self._delete_constraint_sql(self.sql_delete_fk, new_rel.related_model, fk_name))\n        # Removed an index? (no strict check, as multiple indexes are possible)\n        # Remove indexes if db_index switched to False or a unique constraint\n        # will now be used in lieu of an index. The following lines from the\n        # truth table show all True cases; the rest are False:\n        #\n        # old_field.db_index | old_field.unique | new_field.db_index | new_field.unique\n        # ------------------------------------------------------------------------------\n        # True               | False            | False              | False\n        # True               | False            | False              | True\n        # True               | False            | True               | True\n        if old_field.db_index and not old_field.unique and (not new_field.db_index or new_field.unique):\n            # Find the index for this field\n            index_names = self._constraint_names(model, [old_field.column], index=True)\n            for index_name in index_names:\n                self.execute(self._delete_constraint_sql(self.sql_delete_index, model, index_name))\n        # Change check constraints?\n        if old_db_params['check'] != new_db_params['check'] and old_db_params['check']:\n            constraint_names = self._constraint_names(model, [old_field.column], check=True)\n            if strict and len(constraint_names) != 1:\n                raise ValueError(\"Found wrong number (%s) of check constraints for %s.%s\" % (\n                    len(constraint_names),\n                    model._meta.db_table,\n                    old_field.column,\n                ))\n            for constraint_name in constraint_names:\n                self.execute(self._delete_constraint_sql(self.sql_delete_check, model, constraint_name))\n        # Have they renamed the column?\n        if old_field.column != new_field.column:\n            self.execute(self._rename_field_sql(model._meta.db_table, old_field, new_field, new_type))\n        # Next, start accumulating actions to do\n        actions = []\n        null_actions = []\n        post_actions = []\n        # Type change?\n        if old_type != new_type:\n            fragment, other_actions = self._alter_column_type_sql(\n                model._meta.db_table, old_field, new_field, new_type\n            )\n            actions.append(fragment)\n            post_actions.extend(other_actions)\n        # When changing a column NULL constraint to NOT NULL with a given\n        # default value, we need to perform 4 steps:\n        #  1. Add a default for new incoming writes\n        #  2. Update existing NULL rows with new default\n        #  3. Replace NULL constraint with NOT NULL\n        #  4. Drop the default again.\n        # Default change?\n        old_default = self.effective_default(old_field)\n        new_default = self.effective_default(new_field)\n        needs_database_default = (\n            old_field.null and\n            not new_field.null and\n            old_default != new_default and\n            new_default is not None and\n            not self.skip_default(new_field)\n        )\n        if needs_database_default:\n            if self.connection.features.requires_literal_defaults:\n                # Some databases can't take defaults as a parameter (oracle)\n                # If this is the case, the individual schema backend should\n                # implement prepare_default\n                actions.append((\n                    self.sql_alter_column_default % {\n                        \"column\": self.quote_name(new_field.column),\n                        \"type\": new_type,\n                        \"default\": self.prepare_default(new_default),\n                    },\n                    [],\n                ))\n            else:\n                actions.append((\n                    self.sql_alter_column_default % {\n                        \"column\": self.quote_name(new_field.column),\n                        \"type\": new_type,\n                        \"default\": \"%s\",\n                    },\n                    [new_default],\n                ))\n        # Nullability change?\n        if old_field.null != new_field.null:\n            if (self.connection.features.interprets_empty_strings_as_nulls and\n                    new_field.get_internal_type() in (\"CharField\", \"TextField\")):\n                # The field is nullable in the database anyway, leave it alone\n                pass\n            elif new_field.null:\n                null_actions.append((\n                    self.sql_alter_column_null % {\n                        \"column\": self.quote_name(new_field.column),\n                        \"type\": new_type,\n                    },\n                    [],\n                ))\n            else:\n                null_actions.append((\n                    self.sql_alter_column_not_null % {\n                        \"column\": self.quote_name(new_field.column),\n                        \"type\": new_type,\n                    },\n                    [],\n                ))\n        # Only if we have a default and there is a change from NULL to NOT NULL\n        four_way_default_alteration = (\n            new_field.has_default() and\n            (old_field.null and not new_field.null)\n        )\n        if actions or null_actions:\n            if not four_way_default_alteration:\n                # If we don't have to do a 4-way default alteration we can\n                # directly run a (NOT) NULL alteration\n                actions = actions + null_actions\n            # Combine actions together if we can (e.g. postgres)\n            if self.connection.features.supports_combined_alters and actions:\n                sql, params = tuple(zip(*actions))\n                actions = [(\", \".join(sql), sum(params, []))]\n            # Apply those actions\n            for sql, params in actions:\n                self.execute(\n                    self.sql_alter_column % {\n                        \"table\": self.quote_name(model._meta.db_table),\n                        \"changes\": sql,\n                    },\n                    params,\n                )\n            if four_way_default_alteration:\n                # Update existing rows with default value\n                self.execute(\n                    self.sql_update_with_default % {\n                        \"table\": self.quote_name(model._meta.db_table),\n                        \"column\": self.quote_name(new_field.column),\n                        \"default\": \"%s\",\n                    },\n                    [new_default],\n                )\n                # Since we didn't run a NOT NULL change before we need to do it\n                # now\n                for sql, params in null_actions:\n                    self.execute(\n                        self.sql_alter_column % {\n                            \"table\": self.quote_name(model._meta.db_table),\n                            \"changes\": sql,\n                        },\n                        params,\n                    )\n        if post_actions:\n            for sql, params in post_actions:\n                self.execute(sql, params)\n        # Added a unique?\n        if (not old_field.unique and new_field.unique) or (\n            old_field.primary_key and not new_field.primary_key and new_field.unique\n        ):\n            self.execute(self._create_unique_sql(model, [new_field.column]))\n        # Added an index? Add an index if db_index switched to True or a unique\n        # constraint will no longer be used in lieu of an index. The following\n        # lines from the truth table show all True cases; the rest are False:\n        #\n        # old_field.db_index | old_field.unique | new_field.db_index | new_field.unique\n        # ------------------------------------------------------------------------------\n        # False              | False            | True               | False\n        # False              | True             | True               | False\n        # True               | True             | True               | False\n        if (not old_field.db_index or old_field.unique) and new_field.db_index and not new_field.unique:\n            self.execute(self._create_index_sql(model, [new_field]))\n        # Type alteration on primary key? Then we need to alter the column\n        # referring to us.\n        rels_to_update = []\n        if old_field.primary_key and new_field.primary_key and old_type != new_type:\n            rels_to_update.extend(_related_non_m2m_objects(old_field, new_field))\n        # Changed to become primary key?\n        # Note that we don't detect unsetting of a PK, as we assume another field\n        # will always come along and replace it.\n        if not old_field.primary_key and new_field.primary_key:\n            # First, drop the old PK\n            constraint_names = self._constraint_names(model, primary_key=True)\n            if strict and len(constraint_names) != 1:\n                raise ValueError(\"Found wrong number (%s) of PK constraints for %s\" % (\n                    len(constraint_names),\n                    model._meta.db_table,\n                ))\n            for constraint_name in constraint_names:\n                self.execute(self._delete_constraint_sql(self.sql_delete_pk, model, constraint_name))\n            # Make the new one\n            self.execute(\n                self.sql_create_pk % {\n                    \"table\": self.quote_name(model._meta.db_table),\n                    \"name\": self.quote_name(self._create_index_name(model, [new_field.column], suffix=\"_pk\")),\n                    \"columns\": self.quote_name(new_field.column),\n                }\n            )\n            # Update all referencing columns\n            rels_to_update.extend(_related_non_m2m_objects(old_field, new_field))\n        # Handle our type alters on the other end of rels from the PK stuff above\n        for old_rel, new_rel in rels_to_update:\n            rel_db_params = new_rel.field.db_parameters(connection=self.connection)\n            rel_type = rel_db_params['type']\n            fragment, other_actions = self._alter_column_type_sql(\n                new_rel.related_model._meta.db_table, old_rel.field, new_rel.field, rel_type\n            )\n            self.execute(\n                self.sql_alter_column % {\n                    \"table\": self.quote_name(new_rel.related_model._meta.db_table),\n                    \"changes\": fragment[0],\n                },\n                fragment[1],\n            )\n            for sql, params in other_actions:\n                self.execute(sql, params)\n        # Does it have a foreign key?\n        if (new_field.remote_field and\n                (fks_dropped or not old_field.remote_field or not old_field.db_constraint) and\n                new_field.db_constraint):\n            self.execute(self._create_fk_sql(model, new_field, \"_fk_%(to_table)s_%(to_column)s\"))\n        # Rebuild FKs that pointed to us if we previously had to drop them\n        if old_field.primary_key and new_field.primary_key and old_type != new_type:\n            for rel in new_field.model._meta.related_objects:\n                if not rel.many_to_many:\n                    self.execute(self._create_fk_sql(rel.related_model, rel.field, \"_fk\"))\n        # Does it have check constraints we need to add?\n        if old_db_params['check'] != new_db_params['check'] and new_db_params['check']:\n            self.execute(\n                self.sql_create_check % {\n                    \"table\": self.quote_name(model._meta.db_table),\n                    \"name\": self.quote_name(self._create_index_name(model, [new_field.column], suffix=\"_check\")),\n                    \"column\": self.quote_name(new_field.column),\n                    \"check\": new_db_params['check'],\n                }\n            )\n        # Drop the default if we need to\n        # (Django usually does not use in-database defaults)\n        if needs_database_default:\n            sql = self.sql_alter_column % {\n                \"table\": self.quote_name(model._meta.db_table),\n                \"changes\": self.sql_alter_column_no_default % {\n                    \"column\": self.quote_name(new_field.column),\n                    \"type\": new_type,\n                }\n            }\n            self.execute(sql)\n        # Reset connection if required\n        if self.connection.features.connection_persists_old_columns:\n            self.connection.close()"
        },
        {
          "file": "django/db/backends/base/schema.py",
          "type": "function",
          "name": "_delete_constraint_sql",
          "class_name": "BaseDatabaseSchemaEditor",
          "code": "def _delete_constraint_sql(self, template, model, name):\n        return template % {\n            \"table\": self.quote_name(model._meta.db_table),\n            \"name\": self.quote_name(name),\n        }"
        },
        {
          "file": "django/db/backends/base/schema.py",
          "type": "function",
          "name": "_constraint_names",
          "class_name": "BaseDatabaseSchemaEditor",
          "code": "def _constraint_names(self, model, column_names=None, unique=None,\n                          primary_key=None, index=None, foreign_key=None,\n                          check=None):\n        \"\"\"Return all constraint names matching the columns and conditions.\"\"\"\n        if column_names is not None:\n            column_names = [\n                self.connection.introspection.column_name_converter(name)\n                for name in column_names\n            ]\n        with self.connection.cursor() as cursor:\n            constraints = self.connection.introspection.get_constraints(cursor, model._meta.db_table)\n        result = []\n        for name, infodict in constraints.items():\n            if column_names is None or column_names == infodict['columns']:\n                if unique is not None and infodict['unique'] != unique:\n                    continue\n                if primary_key is not None and infodict['primary_key'] != primary_key:\n                    continue\n                if index is not None and infodict['index'] != index:\n                    continue\n                if check is not None and infodict['check'] != check:\n                    continue\n                if foreign_key is not None and not infodict['foreign_key']:\n                    continue\n                result.append(name)\n        return result"
        }
      ]
    },
    {
      "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"
        }
      ]
    },
    {
      "pr_number": 10468,
      "pr_title": "Fixed #29827 -- Fixed reuse of existing test DBs with --keepdb on MySQL.",
      "pr_body": "https://code.djangoproject.com/ticket/29827",
      "issue_id": 29827,
      "issue_title": "cloned DBs are not reused during tests on MySQL",
      "issue_body": "When I'm using\ntests/runtests.py --parallel=8 -k\n, I expect following runs to be faster than the prior one, because test DBs are created once and are reused in the following runs. On MySQL they take the same time. I see\nmysqldump\nin processes every time I run tests.\nIt seems that this regression was introduced in\ne1253bc26facfa1d0fca161f43925e99c2591ced\n.",
      "issue_closed_at": "2018-10-25T18:38:15",
      "base_commit": "76b3367035889d87ffef7a52cd44d70e30537f6f",
      "changes": [
        {
          "file": "django/db/backends/mysql/creation.py",
          "type": "function",
          "name": "sql_table_creation_suffix",
          "class_name": "DatabaseCreation",
          "code": "def sql_table_creation_suffix(self):\n        suffix = []\n        test_settings = self.connection.settings_dict['TEST']\n        if test_settings['CHARSET']:\n            suffix.append('CHARACTER SET %s' % test_settings['CHARSET'])\n        if test_settings['COLLATION']:\n            suffix.append('COLLATE %s' % test_settings['COLLATION'])\n        return ' '.join(suffix)"
        },
        {
          "file": "django/db/backends/mysql/creation.py",
          "type": "function",
          "name": "_clone_test_db",
          "class_name": "DatabaseCreation",
          "code": "def _clone_test_db(self, suffix, verbosity, keepdb=False):\n        source_database_name = self.connection.settings_dict['NAME']\n        target_database_name = self.get_test_db_clone_settings(suffix)['NAME']\n        test_db_params = {\n            'dbname': self.connection.ops.quote_name(target_database_name),\n            'suffix': self.sql_table_creation_suffix(),\n        }\n        with self._nodb_connection.cursor() as cursor:\n            try:\n                self._execute_create_test_db(cursor, test_db_params, keepdb)\n            except Exception:\n                try:\n                    if verbosity >= 1:\n                        self.log('Destroying old test database for alias %s…' % (\n                            self._get_database_display_str(verbosity, target_database_name),\n                        ))\n                    cursor.execute('DROP DATABASE %(dbname)s' % test_db_params)\n                    self._execute_create_test_db(cursor, test_db_params, keepdb)\n                except Exception as e:\n                    self.log('Got an error recreating the test database: %s' % e)\n                    sys.exit(2)\n\n        dump_cmd = DatabaseClient.settings_to_cmd_args(self.connection.settings_dict)\n        dump_cmd[0] = 'mysqldump'\n        dump_cmd[-1] = source_database_name\n        load_cmd = DatabaseClient.settings_to_cmd_args(self.connection.settings_dict)\n        load_cmd[-1] = target_database_name\n\n        with subprocess.Popen(dump_cmd, stdout=subprocess.PIPE) as dump_proc:\n            with subprocess.Popen(load_cmd, stdin=dump_proc.stdout, stdout=subprocess.DEVNULL):\n                # Allow dump_proc to receive a SIGPIPE if the load process exits.\n                dump_proc.stdout.close()"
        },
        {
          "file": "django/db/backends/mysql/creation.py",
          "type": "function",
          "name": "_clone_test_db",
          "class_name": "DatabaseCreation",
          "code": "def _clone_test_db(self, suffix, verbosity, keepdb=False):\n        source_database_name = self.connection.settings_dict['NAME']\n        target_database_name = self.get_test_db_clone_settings(suffix)['NAME']\n        test_db_params = {\n            'dbname': self.connection.ops.quote_name(target_database_name),\n            'suffix': self.sql_table_creation_suffix(),\n        }\n        with self._nodb_connection.cursor() as cursor:\n            try:\n                self._execute_create_test_db(cursor, test_db_params, keepdb)\n            except Exception:\n                try:\n                    if verbosity >= 1:\n                        self.log('Destroying old test database for alias %s…' % (\n                            self._get_database_display_str(verbosity, target_database_name),\n                        ))\n                    cursor.execute('DROP DATABASE %(dbname)s' % test_db_params)\n                    self._execute_create_test_db(cursor, test_db_params, keepdb)\n                except Exception as e:\n                    self.log('Got an error recreating the test database: %s' % e)\n                    sys.exit(2)\n\n        dump_cmd = DatabaseClient.settings_to_cmd_args(self.connection.settings_dict)\n        dump_cmd[0] = 'mysqldump'\n        dump_cmd[-1] = source_database_name\n        load_cmd = DatabaseClient.settings_to_cmd_args(self.connection.settings_dict)\n        load_cmd[-1] = target_database_name\n\n        with subprocess.Popen(dump_cmd, stdout=subprocess.PIPE) as dump_proc:\n            with subprocess.Popen(load_cmd, stdin=dump_proc.stdout, stdout=subprocess.DEVNULL):\n                # Allow dump_proc to receive a SIGPIPE if the load process exits.\n                dump_proc.stdout.close()"
        }
      ]
    },
    {
      "pr_number": 10653,
      "pr_title": "Fixed #29959 -- Cached GEOS version in WKBWriter class",
      "pr_body": "https://code.djangoproject.com/ticket/29959",
      "issue_id": 29959,
      "issue_title": "Random LooseVersion errors while getting multiple wkb values",
      "issue_body": "From\n[f185d929fa1c0caa]\n, each call to geometry\nwkb\nvalue (used in the json representation) will call\nGEOSversion()\nto test a corner case condition.\nUnfortunately and randomly, this causes\n'LooseVersion' object has no attribute 'version'\nerrors. It might be memory related because the same operation sometimes fails, sometimes succeeds in the exact same code conditions.\nIt's hard to know what part of the code to blame, but I would argue that calling a GEOS method for each wkb value retrieval is suboptimal, as the GEOS version should not change between server restart. We may find a way to cache the GEOS version between calls.",
      "issue_closed_at": "2018-11-16T14:12:45",
      "base_commit": "97cec6f75d9d9b86892829f784e5e9dabfd1242a",
      "changes": [
        {
          "file": "django/contrib/gis/geos/prototypes/io.py",
          "type": "class",
          "name": "WKBWriter",
          "code": "class WKBWriter(IOBase):\n    _constructor = wkb_writer_create\n    ptr_type = WKB_WRITE_PTR\n    destructor = wkb_writer_destroy\n\n    def __init__(self, dim=2):\n        super().__init__()\n        self.outdim = dim\n\n    def _handle_empty_point(self, geom):\n        from django.contrib.gis.geos import Point\n        if isinstance(geom, Point) and geom.empty:\n            if self.srid:\n                # PostGIS uses POINT(NaN NaN) for WKB representation of empty\n                # points. Use it for EWKB as it's a PostGIS specific format.\n                # https://trac.osgeo.org/postgis/ticket/3181\n                geom = Point(float('NaN'), float('NaN'), srid=geom.srid)\n            else:\n                raise ValueError('Empty point is not representable in WKB.')\n        return geom\n\n    def write(self, geom):\n        \"Return the WKB representation of the given geometry.\"\n        from django.contrib.gis.geos import Polygon\n        geom = self._handle_empty_point(geom)\n        wkb = wkb_writer_write(self.ptr, geom.ptr, byref(c_size_t()))\n        if geos_version_tuple() < (3, 6, 1) and isinstance(geom, Polygon) and geom.empty:\n            # Fix GEOS output for empty polygon.\n            # See https://trac.osgeo.org/geos/ticket/680.\n            wkb = wkb[:-8] + b'\\0' * 4\n        return memoryview(wkb)\n\n    def write_hex(self, geom):\n        \"Return the HEXEWKB representation of the given geometry.\"\n        from django.contrib.gis.geos.polygon import Polygon\n        geom = self._handle_empty_point(geom)\n        wkb = wkb_writer_write_hex(self.ptr, geom.ptr, byref(c_size_t()))\n        if geos_version_tuple() < (3, 6, 1) and isinstance(geom, Polygon) and geom.empty:\n            wkb = wkb[:-16] + b'0' * 8\n        return wkb\n\n    # ### WKBWriter Properties ###\n\n    # Property for getting/setting the byteorder.\n    def _get_byteorder(self):\n        return wkb_writer_get_byteorder(self.ptr)\n\n    def _set_byteorder(self, order):\n        if order not in (0, 1):\n            raise ValueError('Byte order parameter must be 0 (Big Endian) or 1 (Little Endian).')\n        wkb_writer_set_byteorder(self.ptr, order)\n\n    byteorder = property(_get_byteorder, _set_byteorder)\n\n    # Property for getting/setting the output dimension.\n    @property\n    def outdim(self):\n        return wkb_writer_get_outdim(self.ptr)\n\n    @outdim.setter\n    def outdim(self, new_dim):\n        if new_dim not in (2, 3):\n            raise ValueError('WKB output dimension must be 2 or 3')\n        wkb_writer_set_outdim(self.ptr, new_dim)\n\n    # Property for getting/setting the include srid flag.\n    @property\n    def srid(self):\n        return bool(wkb_writer_get_include_srid(self.ptr))\n\n    @srid.setter\n    def srid(self, include):\n        wkb_writer_set_include_srid(self.ptr, bool(include))"
        },
        {
          "file": "django/contrib/gis/geos/prototypes/io.py",
          "type": "function",
          "name": "write",
          "class_name": "WKBWriter",
          "code": "def write(self, geom):\n        \"Return the WKB representation of the given geometry.\"\n        from django.contrib.gis.geos import Polygon\n        geom = self._handle_empty_point(geom)\n        wkb = wkb_writer_write(self.ptr, geom.ptr, byref(c_size_t()))\n        if geos_version_tuple() < (3, 6, 1) and isinstance(geom, Polygon) and geom.empty:\n            # Fix GEOS output for empty polygon.\n            # See https://trac.osgeo.org/geos/ticket/680.\n            wkb = wkb[:-8] + b'\\0' * 4\n        return memoryview(wkb)"
        },
        {
          "file": "django/contrib/gis/geos/prototypes/io.py",
          "type": "function",
          "name": "write_hex",
          "class_name": "WKBWriter",
          "code": "def write_hex(self, geom):\n        \"Return the HEXEWKB representation of the given geometry.\"\n        from django.contrib.gis.geos.polygon import Polygon\n        geom = self._handle_empty_point(geom)\n        wkb = wkb_writer_write_hex(self.ptr, geom.ptr, byref(c_size_t()))\n        if geos_version_tuple() < (3, 6, 1) and isinstance(geom, Polygon) and geom.empty:\n            wkb = wkb[:-16] + b'0' * 8\n        return wkb"
        }
      ]
    }
  ]
}