{
    "Selected_candidate": {
        "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))"
            }
        ]
    },
    "Justification": "Candidate D directly addresses the issue of dropping `unique_together` constraints, which is central to the CURRENT bug. The bug discusses the challenge of transitioning from a ForeignKey to a ManyToManyField, which involves modifying or removing constraints. Candidate D provides insights into how one can handle unique constraints while migrating, which is similar to the situation the CURRENT bug encounters. As both entail challenges with handling migrations and constraints, this candidate would be the most useful for fixing and debugging the CURRENT bug."
}