{
  "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": "Tested with Postgresql backend, Django versions 3.2 & 4.0\nDefining an Index in the Meta of a model ignores the DEFAULT_INDEX_TABLESPACE setting if multiple fields are used for the index.\nThe setting is honoured if only a single field defines the index.\nExample case, with a postgresql DB defined in settings, as well as:\nDEFAULT_TABLESPACE = 'data_ts'\nDEFAULT_INDEX_TABLESPACE = 'index_ts'\nThe below model definition:\nclass MyModel(models.Model):\n    foo = models.CharField(max_length=10)\n    bar = models.CharField(max_length=10)\n\n    class Meta:\n        indexes = [\n            models.Index(fields=['foo']),\n            models.Index(fields=['foo', 'bar']),\n        ]\ngenerates this output from management command\nsqlmigrate\nrun on the produced migration:\nBEGIN;\n--\n-- Create model MyModel\n--\nCREATE TABLE \"cbcap_mymodel\" (\"id\" bigserial NOT NULL PRIMARY KEY USING INDEX TABLESPACE \"index_ts\", \"foo\" varchar(10) NOT NULL, \"bar\" varchar(10) NOT NULL) TABLESPACE \"data_ts\";\n--\n-- Create index cbcap_mymod_foo_f01529_idx on field(s) foo of model mymodel\n--\nCREATE INDEX \"cbcap_mymod_foo_f01529_idx\" ON \"cbcap_mymodel\" (\"foo\") TABLESPACE \"index_ts\";\n--\n-- Create index cbcap_mymod_foo_44bcd6_idx on field(s) foo, bar of model mymodel\n--\nCREATE INDEX \"cbcap_mymod_foo_44bcd6_idx\" ON \"cbcap_mymodel\" (\"foo\", \"bar\") TABLESPACE \"data_ts\";\nCOMMIT;\nNote the 'data_ts' tablespace is used for the second index",
      "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": "I have an erroneous\nunique_together\nconstraint on a model's primary key (\nunique_together = (('id',),)\n) that cannot be dropped by a migration. Apparently the migration tries to find all unique constraints on the column and expects there to be only one, but I've got two — the primary key and the\nunique_together\nconstraint:\nIndexes:\n    \"foo_bar_pkey\" PRIMARY KEY, btree (id)\n    \"foo_bar_id_1c3b3088c74c3b17_uniq\" UNIQUE CONSTRAINT, btree (id)\nDatabase is PostgreSQL, if that makes any difference.",
      "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": "I am trying to migrate a table from an auto-incrementing Integer PK to a UUID PK.\nI have custom-written migrations that add the new UUID field, add new foreign key fields in dependent tables, populate them, remove the old FKs, etc.\nOne of the final steps of the process is to alter the new uuid field in the table to be a primary_key:\nmigrations\n.\nAlterField\n(\nmodel_name\n=\n'examplelayout'\n,\nname\n=\n'uuid'\n,\nfield\n=\nmodels\n.\nUUIDField\n(\ndefault\n=\nuuid\n.\nuuid4\n,\nprimary_key\n=\nTrue\n,\neditable\n=\nFalse\n),\n),\nThe whole process works great on Postgres, but this specific step ends up in an error on Oracle 12, despite the fact that before this step there are no indexes in the DB on this field.\nquery = 'ALTER TABLE \"VARIO_EXAMPLELAYOUT\" ADD CONSTRAINT \"VARIO_EXA_UUID_8498C9C8_P\" PRIMARY KEY (\"UUID\")'\n...\ndjango.db.utils.DatabaseError: ORA-02261: such unique or primary key already exists in the table\nThe issue seems to be that Django tries to first add a UNIQUE constraint for the field, then a PRIMARY KEY on the same field. Here are all the SQL queries executed by this specific migration step:\n==================== STEP =========================  4  =====================\n\n\n[2018-06-15 08:51:11.141 | DEBUG | django.db.backends.schema:124] ALTER TABLE \"VARIO_EXAMPLELAYOUT\" MODIFY \"UUID\" DEFAULT '9cf3fe16f11d4fa09b62e337c50ae5f7'; (params [])\n[2018-06-15 08:51:11.212 | DEBUG | django.db.backends:111] (0.071) QUERY = 'ALTER TABLE \"VARIO_EXAMPLELAYOUT\" MODIFY \"UUID\" DEFAULT \\'9cf3fe16f11d4fa09b62e337c50ae5f7\\'' - PARAMS = (); args=[]\n[2018-06-15 08:51:11.212 | DEBUG | django.db.backends.schema:124] UPDATE \"VARIO_EXAMPLELAYOUT\" SET \"UUID\" = %s WHERE \"UUID\" IS NULL; (params ['347ac08714bd4c208ffd2f79e7932bb3'])\n[2018-06-15 08:51:11.269 | DEBUG | django.db.backends:111] (0.056) QUERY = 'UPDATE \"VARIO_EXAMPLELAYOUT\" SET \"UUID\" = :arg0 WHERE \"UUID\" IS NULL' - PARAMS = ('347ac08714bd4c208ffd2f79e7932bb3',); args=['347ac08714bd4c208ffd2f79e7932bb3']\n[2018-06-15 08:51:11.269 | DEBUG | django.db.backends.schema:124] ALTER TABLE \"VARIO_EXAMPLELAYOUT\" MODIFY \"UUID\" NOT NULL; (params [])\n[2018-06-15 08:51:11.343 | DEBUG | django.db.backends:111] (0.073) QUERY = 'ALTER TABLE \"VARIO_EXAMPLELAYOUT\" MODIFY \"UUID\" NOT NULL' - PARAMS = (); args=[]\n[2018-06-15 08:51:11.343 | DEBUG | django.db.backends.schema:124] ALTER TABLE \"VARIO_EXAMPLELAYOUT\" ADD CONSTRAINT vario_exa_uuid_8498c9c8_u UNIQUE (\"UUID\"); (params ())\n[2018-06-15 08:51:11.437 | DEBUG | django.db.backends:111] (0.094) QUERY = 'ALTER TABLE \"VARIO_EXAMPLELAYOUT\" ADD CONSTRAINT vario_exa_uuid_8498c9c8_u UNIQUE (\"UUID\")' - PARAMS = (); args=()\n[2018-06-15 08:51:11.792 | DEBUG | django.db.backends:111] (0.354) QUERY = \"\\n            SELECT\\n                user_constraints.constraint_name,\\n                LISTAGG(LOWER(cols.column_name), ',') WITHIN GROUP (ORDER BY cols.position),\\n                CASE user_constraints.constraint_type\\n                    WHEN 'P' THEN 1\\n                    ELSE 0\\n                END AS is_primary_key,\\n                CASE\\n                    WHEN user_constraints.constraint_type IN ('P', 'U') THEN 1\\n                    ELSE 0\\n                END AS is_unique,\\n                CASE user_constraints.constraint_type\\n                    WHEN 'C' THEN 1\\n                    ELSE 0\\n                END AS is_check_constraint\\n            FROM\\n                user_constraints\\n            LEFT OUTER JOIN\\n                user_cons_columns cols ON user_constraints.constraint_name = cols.constraint_name\\n            WHERE\\n                user_constraints.constraint_type = ANY('P', 'U', 'C')\\n                AND user_constraints.table_name = UPPER(:arg0)\\n            GROUP BY user_constraints.constraint_name, user_constraints.constraint_type\\n        \" - PARAMS = ('vario_examplelayout',); args=['vario_examplelayout']\n[2018-06-15 08:51:12.480 | DEBUG | django.db.backends:111] (0.688) QUERY = \"\\n            SELECT\\n                cons.constraint_name,\\n                LISTAGG(LOWER(cols.column_name), ',') WITHIN GROUP (ORDER BY cols.position),\\n                LOWER(rcols.table_name),\\n                LOWER(rcols.column_name)\\n            FROM\\n                user_constraints cons\\n            INNER JOIN\\n                user_cons_columns rcols ON rcols.constraint_name = cons.r_constraint_name AND rcols.position = 1\\n            LEFT OUTER JOIN\\n                user_cons_columns cols ON cons.constraint_name = cols.constraint_name\\n            WHERE\\n                cons.constraint_type = 'R' AND\\n                cons.table_name = UPPER(:arg0)\\n            GROUP BY cons.constraint_name, rcols.table_name, rcols.column_name\\n        \" - PARAMS = ('vario_examplelayout',); args=['vario_examplelayout']\n[2018-06-15 08:51:13.695 | DEBUG | django.db.backends:111] (1.214) QUERY = \"\\n            SELECT\\n                ind.index_name,\\n                LOWER(ind.index_type),\\n                LISTAGG(LOWER(cols.column_name), ',') WITHIN GROUP (ORDER BY cols.column_position),\\n                LISTAGG(cols.descend, ',') WITHIN GROUP (ORDER BY cols.column_position)\\n            FROM\\n                user_ind_columns cols, user_indexes ind\\n            WHERE\\n                cols.table_name = UPPER(:arg0) AND\\n                NOT EXISTS (\\n                    SELECT 1\\n                    FROM user_constraints cons\\n                    WHERE ind.index_name = cons.index_name\\n                ) AND cols.index_name = ind.index_name\\n            GROUP BY ind.index_name, ind.index_type\\n        \" - PARAMS = ('vario_examplelayout',); args=['vario_examplelayout']\n[2018-06-15 08:51:13.696 | DEBUG | django.db.backends.schema:124] ALTER TABLE \"VARIO_EXAMPLELAYOUT\" ADD CONSTRAINT \"VARIO_EXA_UUID_8498C9C8_P\" PRIMARY KEY (\"UUID\"); (params ())\n[2018-06-15 08:51:13.775 | DEBUG | django.db.backends:111] (0.079) QUERY = 'ALTER TABLE \"VARIO_EXAMPLELAYOUT\" ADD CONSTRAINT \"VARIO_EXA_UUID_8498C9C8_P\" PRIMARY KEY (\"UUID\")' - PARAMS = (); args=()\nTraceback (most recent call last):\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/backends/utils.py\", line 85, in _execute\n    return self.cursor.execute(sql, params)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/backends/oracle/base.py\", line 500, in execute\n    return self.cursor.execute(query, self._param_generator(params))\ncx_Oracle.DatabaseError: ORA-02261: such unique or primary key already exists in the table\n\nThe above exception was the direct cause of the following exception:\n\nTraceback (most recent call last):\n  File \"./manage.py\", line 35, in <module>\n    execute_from_command_line(sys.argv)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/core/management/__init__.py\", line 371, in execute_from_command_line\n    utility.execute()\n  File \"/var/www/venv/lib/python3.5/site-packages/django/core/management/__init__.py\", line 365, in execute\n    self.fetch_command(subcommand).run_from_argv(self.argv)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/core/management/base.py\", line 288, in run_from_argv\n    self.execute(*args, **cmd_options)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/core/management/base.py\", line 335, in execute\n    output = self.handle(*args, **options)\n  File \"/var/www/forms/forms/common/management/commands/migrate.py\", line 54, in handle\n    super().handle(*args, **options)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/core/management/commands/migrate.py\", line 200, in handle\n    fake_initial=fake_initial,\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/migrations/executor.py\", line 117, in migrate\n    state = self._migrate_all_forwards(state, plan, full_plan, fake=fake, fake_initial=fake_initial)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/migrations/executor.py\", line 147, in _migrate_all_forwards\n    state = self.apply_migration(state, migration, fake=fake, fake_initial=fake_initial)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/migrations/executor.py\", line 244, in apply_migration\n    state = migration.apply(state, schema_editor)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/migrations/migration.py\", line 122, in apply\n    operation.database_forwards(self.app_label, schema_editor, old_state, project_state)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/migrations/operations/fields.py\", line 216, in database_forwards\n    schema_editor.alter_field(from_model, from_field, to_field)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/backends/oracle/schema.py\", line 59, in alter_field\n    super().alter_field(model, old_field, new_field, strict)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/backends/base/schema.py\", line 525, in alter_field\n    old_db_params, new_db_params, strict)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/backends/base/schema.py\", line 725, in _alter_field\n    \"columns\": self.quote_name(new_field.column),\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/backends/base/schema.py\", line 133, in execute\n    cursor.execute(sql, params)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/backends/utils.py\", line 100, in execute\n    return super().execute(sql, params)\n  File \"/var/www/venv/lib/python3.5/site-packages/raven/contrib/django/client.py\", line 127, in execute\n    return real_execute(self, sql, params)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/backends/utils.py\", line 68, in execute\n    return self._execute_with_wrappers(sql, params, many=False, executor=self._execute)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/backends/utils.py\", line 77, in _execute_with_wrappers\n    return executor(sql, params, many, context)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/backends/utils.py\", line 85, in _execute\n    return self.cursor.execute(sql, params)\n  File \"/var/www/forms/forms/forms/db/backends/utils.py\", line 16, in execute\n    return super().execute(sql, params)\n  File \"/var/www/venv/lib/python3.5/site-packages/raven/contrib/django/client.py\", line 127, in execute\n    return real_execute(self, sql, params)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/backends/utils.py\", line 68, in execute\n    return self._execute_with_wrappers(sql, params, many=False, executor=self._execute)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/backends/utils.py\", line 77, in _execute_with_wrappers\n    return executor(sql, params, many, context)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/backends/utils.py\", line 85, in _execute\n    return self.cursor.execute(sql, params)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/utils.py\", line 89, in __exit__\n    raise dj_exc_value.with_traceback(traceback) from exc_value\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/backends/utils.py\", line 85, in _execute\n    return self.cursor.execute(sql, params)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/backends/oracle/base.py\", line 500, in execute\n    return self.cursor.execute(query, self._param_generator(params))\ndjango.db.utils.DatabaseError: ORA-02261: such unique or primary key already exists in the table\nNotice the two calls:\n[2018-06-15 08:51:11.437 | DEBUG | django.db.backends:111] (0.094) QUERY = 'ALTER TABLE \"VARIO_EXAMPLELAYOUT\" ADD CONSTRAINT vario_exa_uuid_8498c9c8_u UNIQUE (\"UUID\")' - PARAMS = (); args=()\nand a bit below:\n[2018-06-15 08:51:13.775 | DEBUG | django.db.backends:111] (0.079) QUERY = 'ALTER TABLE \"VARIO_EXAMPLELAYOUT\" ADD CONSTRAINT \"VARIO_EXA_UUID_8498C9C8_P\" PRIMARY KEY (\"UUID\")' - PARAMS = (); args=()\n...which raises the error\nThe only other operations before this on this field that I have in one of the previous migrations is to actually add this UUID field as a nullable, non-unique field, and then in a data migration to populate it with a unique uuids for all ExampleLayouts:\n# prepare to migrate to UUID PKs for ExampleLayout and LayoutField\n        migrations.AddField(\n            model_name='examplelayout',\n            name='uuid',\n            field=models.UUIDField(null=True, default=uuid.uuid4),\n        ),\nFor completeness, the initial models look like this:\nclass\nExampleLayout\n(\nmodels\n.\nModel\n):\nimage\n=\nmodels\n.\nOneToOneField\n(\nImage\n,\non_delete\n=\nmodels\n.\nCASCADE\n,\nrelated_name\n=\n'example_layout'\n)\ndoc_type\n=\nmodels\n.\nForeignKey\n(\nDocumentType\n,\non_delete\n=\nmodels\n.\nCASCADE\n,\nrelated_name\n=\n'layouts'\n)\nclass\nLayoutField\n(\nmodels\n.\nModel\n):\ncore_field_type\n=\nmodels\n.\nForeignKey\n(\nCoreFieldType\n,\non_delete\n=\nmodels\n.\nCASCADE\n,\nrelated_name\n=\n'layout_fields'\n)\nlayout\n=\nmodels\n.\nForeignKey\n(\nExampleLayout\n,\non_delete\n=\nmodels\n.\nCASCADE\n,\nrelated_name\n=\n'fields'\n)\nbounding_box\n=\nBoundingBoxCharField\n()\ntranscription\n=\nmodels\n.\nCharField\n(\nmax_length\n=\n128\n)\nwhile their final \"target\" version that I am trying to migrate to is this:\nclass\nExampleLayout\n(\nmodels\n.\nModel\n):\nuuid\n=\nmodels\n.\nUUIDField\n(\ndefault\n=\nuuid\n.\nuuid4\n,\nprimary_key\n=\nTrue\n,\neditable\n=\nFalse\n)\nimage\n=\nmodels\n.\nOneToOneField\n(\nImage\n,\non_delete\n=\nmodels\n.\nCASCADE\n,\nrelated_name\n=\n'example_layout'\n)\ndoc_type\n=\nmodels\n.\nForeignKey\n(\nDocumentType\n,\non_delete\n=\nmodels\n.\nCASCADE\n,\nrelated_name\n=\n'layouts'\n)\nclass\nLayoutField\n(\nmodels\n.\nModel\n):\nuuid\n=\nmodels\n.\nUUIDField\n(\ndefault\n=\nuuid\n.\nuuid4\n,\nprimary_key\n=\nTrue\n,\neditable\n=\nFalse\n)\ncore_field_type\n=\nmodels\n.\nForeignKey\n(\nCoreFieldType\n,\non_delete\n=\nmodels\n.\nCASCADE\n,\nrelated_name\n=\n'layout_fields'\n)\nlayout\n=\nmodels\n.\nForeignKey\n(\nExampleLayout\n,\non_delete\n=\nmodels\n.\nCASCADE\n,\nrelated_name\n=\n'fields'\n)\nbounding_box\n=\nBoundingBoxCharField\n()\ntranscription\n=\nmodels\n.\nCharField\n(\nmax_length\n=\n128\n)",
      "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": "I think this results in a missing index for GIS fields on apps that have migrations.",
      "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": "BrinIndex.suffix ( django.contrib.postgres.indexes) longer 3 char, but Index.set_name_with_model method allowed 3 char in suffix\nAssertionError: Index too long for multiple database support. Is self.suffix longer than 3 characters?",
      "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"
        }
      ]
    }
  ]
}