{
  "instance_id": "django__django-16046",
  "repo": "django/django",
  "created_at": "2022-09-10T13:27:38Z",
  "problem_statement": "Fix numberformat.py \"string index out of range\" when null\nDescription\n\t\nWhen:\nif str_number[0] == \"-\"\nencounters a number field that's null when formatting for the admin list_display this causes an \nIndexError: string index out of range\nI can attach the proposed fix here, or open a pull request on GitHub if you like?\n",
  "patch": "diff --git a/django/utils/numberformat.py b/django/utils/numberformat.py\n--- a/django/utils/numberformat.py\n+++ b/django/utils/numberformat.py\n@@ -25,6 +25,8 @@ def format(\n         module in locale.localeconv() LC_NUMERIC grouping (e.g. (3, 2, 0)).\n     * thousand_sep: Thousand separator symbol (for example \",\")\n     \"\"\"\n+    if number is None or number == \"\":\n+        return mark_safe(number)\n     use_grouping = (\n         use_l10n or (use_l10n is None and settings.USE_L10N)\n     ) and settings.USE_THOUSAND_SEPARATOR\n",
  "similar_bug_items": [
    {
      "pr_number": 15770,
      "pr_title": "Fixed #33773 -- Made Index with multiple fields respect DEFAULT_INDEX_TABLESPACE.",
      "pr_body": "Added check for the setting in schema's _get_index_tablespace_sql\r\nbefore it falls back to the model's db_tablespace.\r\nThanks to Simon Charette for locating where issue lay.",
      "issue_id": 33773,
      "issue_title": "DEFAULT_INDEX_TABLESPACE setting is ignored for Indexes defined with multiple fields.",
      "issue_body": "",
      "issue_closed_at": "2022-06-13T01:46:58",
      "base_commit": "4996eaa7b5b11f822d1c97df79644738df3fed68",
      "changes": [
        {
          "file": "django/db/backends/base/schema.py",
          "type": "line",
          "name": "line 2",
          "code": "import operator\nfrom datetime import datetime\n\nfrom django.db.backends.ddl_references import (\n    Columns,\n    Expressions,"
        },
        {
          "file": "django/db/backends/base/schema.py",
          "type": "function",
          "name": "_get_index_tablespace_sql",
          "class_name": "BaseDatabaseSchemaEditor",
          "code": "def _get_index_tablespace_sql(self, model, fields, db_tablespace=None):\n        if db_tablespace is None:\n            if len(fields) == 1 and fields[0].db_tablespace:\n                db_tablespace = fields[0].db_tablespace\n            elif model._meta.db_tablespace:\n                db_tablespace = model._meta.db_tablespace\n        if db_tablespace is not None:\n            return \" \" + self.connection.ops.tablespace_sql(db_tablespace)\n        return \"\""
        }
      ]
    },
    {
      "pr_number": 15732,
      "pr_title": "Fixed #23740 -- Fixed removing unique_together constraint if exists primary key/unique constraint on the same field.",
      "pr_body": "Hi,\r\n\r\nTackling [ticket-23740](https://code.djangoproject.com/ticket/23740).\r\n\r\nIn short:\r\n\r\n1/ If the field is the primary key, fix it by filtering on `\"primary_key\": False` in `django.db.backends.base.schema.BaseDatabaseSchemaEditor.alter_unique_together`\r\n\r\n2/ If the field has a `unique=True` constraint, to fix we check that the constraint name is the default one we would give to a `unique_together` constraint, based on the observation that a `unique=True` and a `unique_together` constraints are named differently at the time of writing",
      "issue_id": 23740,
      "issue_title": "Cannot drop unique_together constraint on a single field with its own unique=True constraint",
      "issue_body": "",
      "issue_closed_at": "2022-05-26T02:06:57",
      "base_commit": "ce69e34bd646558bb44ea92cecfd98b345a0b3e0",
      "changes": [
        {
          "file": "django/db/backends/base/schema.py",
          "type": "function",
          "name": "alter_unique_together",
          "class_name": "BaseDatabaseSchemaEditor",
          "code": "def alter_unique_together(self, model, old_unique_together, new_unique_together):\n        \"\"\"\n        Deal with a model changing its unique_together. The input\n        unique_togethers must be doubly-nested, not the single-nested\n        [\"foo\", \"bar\"] format.\n        \"\"\"\n        olds = {tuple(fields) for fields in old_unique_together}\n        news = {tuple(fields) for fields in new_unique_together}\n        # Deleted uniques\n        for fields in olds.difference(news):\n            self._delete_composed_index(\n                model, fields, {\"unique\": True}, self.sql_delete_unique\n            )\n        # Created uniques\n        for field_names in news.difference(olds):\n            fields = [model._meta.get_field(field) for field in field_names]\n            self.execute(self._create_unique_sql(model, fields))"
        },
        {
          "file": "django/db/backends/base/schema.py",
          "type": "function",
          "name": "_delete_composed_index",
          "class_name": "BaseDatabaseSchemaEditor",
          "code": "def _delete_composed_index(self, model, fields, constraint_kwargs, sql):\n        meta_constraint_names = {\n            constraint.name for constraint in model._meta.constraints\n        }\n        meta_index_names = {constraint.name for constraint in model._meta.indexes}\n        columns = [model._meta.get_field(field).column for field in fields]\n        constraint_names = self._constraint_names(\n            model,\n            columns,\n            exclude=meta_constraint_names | meta_index_names,\n            **constraint_kwargs,\n        )\n        if len(constraint_names) != 1:\n            raise ValueError(\n                \"Found wrong number (%s) of constraints for %s(%s)\"\n                % (\n                    len(constraint_names),\n                    model._meta.db_table,\n                    \", \".join(columns),\n                )\n            )\n        self.execute(self._delete_constraint_sql(sql, model, constraint_names[0]))"
        },
        {
          "file": "django/db/backends/base/schema.py",
          "type": "function",
          "name": "_create_unique_sql",
          "class_name": "BaseDatabaseSchemaEditor",
          "code": "def _create_unique_sql(\n        self,\n        model,\n        fields,\n        name=None,\n        condition=None,\n        deferrable=None,\n        include=None,\n        opclasses=None,\n        expressions=None,\n    ):\n        if (\n            (\n                deferrable\n                and not self.connection.features.supports_deferrable_unique_constraints\n            )\n            or (condition and not self.connection.features.supports_partial_indexes)\n            or (include and not self.connection.features.supports_covering_indexes)\n            or (\n                expressions and not self.connection.features.supports_expression_indexes\n            )\n        ):\n            return None\n\n        def create_unique_name(*args, **kwargs):\n            return self.quote_name(self._create_index_name(*args, **kwargs))\n\n        compiler = Query(model, alias_cols=False).get_compiler(\n            connection=self.connection\n        )\n        table = model._meta.db_table\n        columns = [field.column for field in fields]\n        if name is None:\n            name = IndexName(table, columns, \"_uniq\", create_unique_name)\n        else:\n            name = self.quote_name(name)\n        if condition or include or opclasses or expressions:\n            sql = self.sql_create_unique_index\n        else:\n            sql = self.sql_create_unique\n        if columns:\n            columns = self._index_columns(\n                table, columns, col_suffixes=(), opclasses=opclasses\n            )\n        else:\n            columns = Expressions(table, expressions, compiler, self.quote_value)\n        return Statement(\n            sql,\n            table=Table(table, self.quote_name),\n            name=name,\n            columns=columns,\n            condition=self._index_condition_sql(condition),\n            deferrable=self._deferrable_constraint_sql(deferrable),\n            include=self._index_include_sql(model, include),\n        )"
        },
        {
          "file": "django/db/backends/base/schema.py",
          "type": "function",
          "name": "create_unique_name",
          "class_name": "BaseDatabaseSchemaEditor",
          "code": "def create_unique_name(*args, **kwargs):\n            return self.quote_name(self._create_index_name(*args, **kwargs))"
        }
      ]
    },
    {
      "pr_number": 10058,
      "pr_title": "Fixed #29496 -- Fixed crash on Oracle when converting a non-unique field to primary key.",
      "pr_body": "Ticket [29496](https://code.djangoproject.com/ticket/29496).",
      "issue_id": 29496,
      "issue_title": "Altering non-unique field to primary key fails on Oracle",
      "issue_body": "",
      "issue_closed_at": "2018-06-17T01:51:24",
      "base_commit": "e95008f2411d50929873f634c3e14ebac811fd28",
      "changes": [
        {
          "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, primary_key=False)\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 the field is a primary key or unique,\n        # which might be a to_field target, and things are going to change.\n        drop_foreign_keys = (\n            (\n                (old_field.primary_key and new_field.primary_key) or\n                (old_field.unique and new_field.unique)\n            ) and old_type != new_type\n        )\n        if drop_foreign_keys:\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            meta_index_names = {index.name for index in model._meta.indexes}\n            # Retrieve only BTREE indexes since this is what's created with\n            # db_index=True.\n            index_names = self._constraint_names(model, [old_field.column], index=True, type_=Index.suffix)\n            for index_name in index_names:\n                if index_name not in meta_index_names:\n                    # The only way to check if an index was created with\n                    # db_index=True or with Index(['field'], name='foo')\n                    # is to look at its name (refs #28053).\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            # Rename all references to the renamed column.\n            for sql in self.deferred_sql:\n                if isinstance(sql, Statement):\n                    sql.rename_column_references(model._meta.db_table, old_field.column, new_field.column)\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(model, old_field, new_field, new_type)\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            actions.append(self._alter_column_default_sql(model, old_field, new_field))\n        # Nullability change?\n        if old_field.null != new_field.null:\n            fragment = self._alter_column_null_sql(model, old_field, new_field)\n            if fragment:\n                null_actions.append(fragment)\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        # If primary_key changed to False, delete the primary key constraint.\n        if old_field.primary_key and not new_field.primary_key:\n            self._delete_primary_key(model, strict)\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        if not old_field.primary_key and new_field.primary_key:\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(\n                        self._create_index_name(model._meta.db_table, [new_field.column], suffix=\"_pk\")\n                    ),\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, 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 drop_foreign_keys:\n            for rel in new_field.model._meta.related_objects:\n                if _is_relevant_relation(rel, new_field) and rel.field.db_constraint:\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(\n                        self._create_index_name(model._meta.db_table, [new_field.column], suffix=\"_check\")\n                    ),\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            changes_sql, params = self._alter_column_default_sql(model, old_field, new_field, drop=True)\n            sql = self.sql_alter_column % {\n                \"table\": self.quote_name(model._meta.db_table),\n                \"changes\": changes_sql,\n            }\n            self.execute(sql, params)\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": "_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, primary_key=False)\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 the field is a primary key or unique,\n        # which might be a to_field target, and things are going to change.\n        drop_foreign_keys = (\n            (\n                (old_field.primary_key and new_field.primary_key) or\n                (old_field.unique and new_field.unique)\n            ) and old_type != new_type\n        )\n        if drop_foreign_keys:\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            meta_index_names = {index.name for index in model._meta.indexes}\n            # Retrieve only BTREE indexes since this is what's created with\n            # db_index=True.\n            index_names = self._constraint_names(model, [old_field.column], index=True, type_=Index.suffix)\n            for index_name in index_names:\n                if index_name not in meta_index_names:\n                    # The only way to check if an index was created with\n                    # db_index=True or with Index(['field'], name='foo')\n                    # is to look at its name (refs #28053).\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            # Rename all references to the renamed column.\n            for sql in self.deferred_sql:\n                if isinstance(sql, Statement):\n                    sql.rename_column_references(model._meta.db_table, old_field.column, new_field.column)\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(model, old_field, new_field, new_type)\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            actions.append(self._alter_column_default_sql(model, old_field, new_field))\n        # Nullability change?\n        if old_field.null != new_field.null:\n            fragment = self._alter_column_null_sql(model, old_field, new_field)\n            if fragment:\n                null_actions.append(fragment)\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        # If primary_key changed to False, delete the primary key constraint.\n        if old_field.primary_key and not new_field.primary_key:\n            self._delete_primary_key(model, strict)\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        if not old_field.primary_key and new_field.primary_key:\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(\n                        self._create_index_name(model._meta.db_table, [new_field.column], suffix=\"_pk\")\n                    ),\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, 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 drop_foreign_keys:\n            for rel in new_field.model._meta.related_objects:\n                if _is_relevant_relation(rel, new_field) and rel.field.db_constraint:\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(\n                        self._create_index_name(model._meta.db_table, [new_field.column], suffix=\"_check\")\n                    ),\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            changes_sql, params = self._alter_column_default_sql(model, old_field, new_field, drop=True)\n            sql = self.sql_alter_column % {\n                \"table\": self.quote_name(model._meta.db_table),\n                \"changes\": changes_sql,\n            }\n            self.execute(sql, params)\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": "_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, primary_key=False)\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 the field is a primary key or unique,\n        # which might be a to_field target, and things are going to change.\n        drop_foreign_keys = (\n            (\n                (old_field.primary_key and new_field.primary_key) or\n                (old_field.unique and new_field.unique)\n            ) and old_type != new_type\n        )\n        if drop_foreign_keys:\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            meta_index_names = {index.name for index in model._meta.indexes}\n            # Retrieve only BTREE indexes since this is what's created with\n            # db_index=True.\n            index_names = self._constraint_names(model, [old_field.column], index=True, type_=Index.suffix)\n            for index_name in index_names:\n                if index_name not in meta_index_names:\n                    # The only way to check if an index was created with\n                    # db_index=True or with Index(['field'], name='foo')\n                    # is to look at its name (refs #28053).\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            # Rename all references to the renamed column.\n            for sql in self.deferred_sql:\n                if isinstance(sql, Statement):\n                    sql.rename_column_references(model._meta.db_table, old_field.column, new_field.column)\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(model, old_field, new_field, new_type)\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            actions.append(self._alter_column_default_sql(model, old_field, new_field))\n        # Nullability change?\n        if old_field.null != new_field.null:\n            fragment = self._alter_column_null_sql(model, old_field, new_field)\n            if fragment:\n                null_actions.append(fragment)\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        # If primary_key changed to False, delete the primary key constraint.\n        if old_field.primary_key and not new_field.primary_key:\n            self._delete_primary_key(model, strict)\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        if not old_field.primary_key and new_field.primary_key:\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(\n                        self._create_index_name(model._meta.db_table, [new_field.column], suffix=\"_pk\")\n                    ),\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, 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 drop_foreign_keys:\n            for rel in new_field.model._meta.related_objects:\n                if _is_relevant_relation(rel, new_field) and rel.field.db_constraint:\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(\n                        self._create_index_name(model._meta.db_table, [new_field.column], suffix=\"_check\")\n                    ),\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            changes_sql, params = self._alter_column_default_sql(model, old_field, new_field, drop=True)\n            sql = self.sql_alter_column % {\n                \"table\": self.quote_name(model._meta.db_table),\n                \"changes\": changes_sql,\n            }\n            self.execute(sql, params)\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": "_field_indexes_sql",
          "class_name": "BaseDatabaseSchemaEditor",
          "code": "def _field_indexes_sql(self, model, field):\n        \"\"\"\n        Return a list of all index SQL statements for the specified field.\n        \"\"\"\n        output = []\n        if self._field_should_be_indexed(model, field):\n            output.append(self._create_index_sql(model, [field]))\n        return output"
        },
        {
          "file": "django/db/backends/oracle/schema.py",
          "type": "function",
          "name": "_field_should_be_indexed",
          "class_name": "DatabaseSchemaEditor",
          "code": "def _field_should_be_indexed(self, model, field):\n        create_index = super()._field_should_be_indexed(model, field)\n        db_type = field.db_type(self.connection)\n        if db_type is not None and db_type.lower() in self.connection._limited_data_types:\n            return False\n        return create_index"
        }
      ]
    },
    {
      "pr_number": 3265,
      "pr_title": "Fixed #23538 -- Added SchemaEditor for MySQL GIS.",
      "pr_body": "",
      "issue_id": 23538,
      "issue_title": "MySQL GIS backend missing SchemaEditor",
      "issue_body": "",
      "issue_closed_at": "2014-09-25T12:53:52",
      "base_commit": "215aa4f53b6bbd07d5c1eecfa94e7fcd00da813e",
      "changes": [
        {
          "file": "django/contrib/gis/db/backends/mysql/base.py",
          "type": "line",
          "name": "line 6",
          "code": "from django.contrib.gis.db.backends.mysql.creation import MySQLCreation\nfrom django.contrib.gis.db.backends.mysql.introspection import MySQLIntrospection\nfrom django.contrib.gis.db.backends.mysql.operations import MySQLOperations\n\n\nclass DatabaseFeatures(BaseSpatialFeatures, MySQLDatabaseFeatures):"
        },
        {
          "file": "django/contrib/gis/db/backends/mysql/base.py",
          "type": "function",
          "name": "__init__",
          "class_name": "DatabaseWrapper",
          "code": "def __init__(self, *args, **kwargs):\n        super(DatabaseWrapper, self).__init__(*args, **kwargs)\n        self.features = DatabaseFeatures(self)\n        self.creation = MySQLCreation(self)\n        self.ops = MySQLOperations(self)\n        self.introspection = MySQLIntrospection(self)"
        },
        {
          "file": "django/contrib/gis/db/backends/mysql/introspection.py",
          "type": "function",
          "name": "get_geometry_type",
          "class_name": "MySQLIntrospection",
          "code": "def get_geometry_type(self, table_name, geo_col):\n        cursor = self.connection.cursor()\n        try:\n            # In order to get the specific geometry type of the field,\n            # we introspect on the table definition using `DESCRIBE`.\n            cursor.execute('DESCRIBE %s' %\n                           self.connection.ops.quote_name(table_name))\n            # Increment over description info until we get to the geometry\n            # column.\n            for column, typ, null, key, default, extra in cursor.fetchall():\n                if column == geo_col:\n                    # Using OGRGeomType to convert from OGC name to Django field.\n                    # MySQL does not support 3D or SRIDs, so the field params\n                    # are empty.\n                    field_type = OGRGeomType(typ).django\n                    field_params = {}\n                    break\n        finally:\n            cursor.close()\n\n        return field_type, field_params"
        },
        {
          "file": "django/contrib/gis/db/backends/spatialite/introspection.py",
          "type": "function",
          "name": "get_geometry_type",
          "class_name": "SpatiaLiteIntrospection",
          "code": "def get_geometry_type(self, table_name, geo_col):\n        cursor = self.connection.cursor()\n        try:\n            # Querying the `geometry_columns` table to get additional metadata.\n            type_col = 'type' if self.connection.ops.spatial_version < (4, 0, 0) else 'geometry_type'\n            cursor.execute('SELECT coord_dimension, srid, %s '\n                           'FROM geometry_columns '\n                           'WHERE f_table_name=%%s AND f_geometry_column=%%s' % type_col,\n                           (table_name, geo_col))\n            row = cursor.fetchone()\n            if not row:\n                raise Exception('Could not find a geometry column for \"%s\".\"%s\"' %\n                                (table_name, geo_col))\n\n            # OGRGeomType does not require GDAL and makes it easy to convert\n            # from OGC geom type name to Django field.\n            field_type = OGRGeomType(row[2]).django\n\n            # Getting any GeometryField keyword arguments that are not the default.\n            dim = row[0]\n            srid = row[1]\n            field_params = {}\n            if srid != 4326:\n                field_params['srid'] = srid\n            if isinstance(dim, six.string_types) and 'Z' in dim:\n                field_params['dim'] = 3\n        finally:\n            cursor.close()\n\n        return field_type, field_params"
        },
        {
          "file": "django/db/backends/mysql/introspection.py",
          "type": "function",
          "name": "get_indexes",
          "class_name": "DatabaseIntrospection",
          "code": "def get_indexes(self, cursor, table_name):\n        cursor.execute(\"SHOW INDEX FROM %s\" % self.connection.ops.quote_name(table_name))\n        # Do a two-pass search for indexes: on first pass check which indexes\n        # are multicolumn, on second pass check which single-column indexes\n        # are present.\n        rows = list(cursor.fetchall())\n        multicol_indexes = set()\n        for row in rows:\n            if row[3] > 1:\n                multicol_indexes.add(row[2])\n        indexes = {}\n        for row in rows:\n            if row[2] in multicol_indexes:\n                continue\n            if row[4] not in indexes:\n                indexes[row[4]] = {'primary_key': False, 'unique': False}\n            # It's possible to have the unique and PK constraints in separate indexes.\n            if row[2] == 'PRIMARY':\n                indexes[row[4]]['primary_key'] = True\n            if not row[1]:\n                indexes[row[4]]['unique'] = True\n        return indexes"
        }
      ]
    },
    {
      "pr_number": 8213,
      "pr_title": "Refs #27935 -- Fixed BrinIndex.max_name_length if a project's default database isn't PostgreSQL.",
      "pr_body": "",
      "issue_id": 27935,
      "issue_title": "BrinIndex crashes if name  > 30 characters",
      "issue_body": "",
      "issue_closed_at": "2017-03-20T10:30:05",
      "base_commit": "6cb0a3ac2836c044987f7a0cca0d1e99d52e002e",
      "changes": [
        {
          "file": "django/contrib/postgres/indexes.py",
          "type": "line",
          "name": "line 1",
          "code": "from django.db import connection\nfrom django.db.models import Index\nfrom django.utils.functional import cached_property\n\n__all__ = ['BrinIndex', 'GinIndex']\n\n\nclass BrinIndex(Index):\n    suffix = 'brin'\n\n    def __init__(self, fields=[], name=None, pages_per_range=None):\n        if pages_per_range is not None and pages_per_range <= 0:"
        },
        {
          "file": "django/contrib/postgres/indexes.py",
          "type": "function",
          "name": "get_sql_create_template_values",
          "class_name": "BrinIndex",
          "code": "def get_sql_create_template_values(self, model, schema_editor, using):\n        parameters = super().get_sql_create_template_values(model, schema_editor, using=' USING brin')\n        if self.pages_per_range is not None:\n            parameters['extra'] = ' WITH (pages_per_range={})'.format(\n                schema_editor.quote_value(self.pages_per_range)) + parameters['extra']\n        return parameters"
        }
      ]
    }
  ]
}