{
  "Selected_candidate": {
    "pr_number": 12477,
    "pr_title": "Fixed #31185 -- Fixed detecting of unique fields in ForeignKey/ForeignObject checks when using Meta.constraints.",
    "pr_body": "Fixed #31185 -- Adjusted model check for UniqueConstraints without conditions\r\n.\r\nChanged _check_unique_target to take into account UniqueConstraints without conditions. Added test in invalid_models_tests, and changed code to take into account pre-existing method total_unique_constraints\r\nTicket number is [31185](https://code.djangoproject.com/ticket/31185)",
    "issue_id": 31185,
    "issue_title": "fields.E310-E311 should take into account UniqueConstraints without conditions.",
    "issue_body": "Hello,\nI'm trying to create migration with this kind of model.\nclass AppUsers(models.Model):\n    name = models.CharField(...)\n    uid = models.CharField(...)\n    source = models.ForeignKey(...)\n\n    class Meta:\n        constraints = [models.UniqueConstraint(fields=['uid', 'source'], name='appusers_uniqueness')]\nWhen I start\nmakemigrations\ncommand in manage.py I've faced fields.E310\n​\nhttps://docs.djangoproject.com/en/2.2/ref/checks/#related-fields\nerror\nIt says that I should add unique_together field in Meta options:\napp_name.AppUsers.field: (fields.E310) No subset of the fields 'uid', 'source' on model 'AppUsers' is unique.\nHINT: Add unique=True on any of those fields or add at least a subset of them to a unique_together constraint.\nIf I change Meta options to unique_together constraint migration passes with no errors.\nclass AppUsers(models.Model):\n    name = models.CharField(...)\n    uid = models.CharField(...)\n    source = models.ForeignKey(...)\n\n    class Meta:\n        unique_together = [['uid', 'source']]\nAs mentioned in docs\n​\nhttps://docs.djangoproject.com/en/2.2/ref/models/options/#unique-together\nunique_together\nmay be deprecated in the future. So I think nobody wants to face this issue when this will be deprecated :)\nThanks,\nPavel",
    "issue_closed_at": "2020-02-27T04:33:35",
    "base_commit": "41ebe60728a15aa273f4d70de92f5246a89c3d4e",
    "changes": [
      {
        "file": "django/db/models/fields/related.py",
        "type": "function",
        "name": "_check_unique_target",
        "class_name": "ForeignObject",
        "code": "def _check_unique_target(self):\n        rel_is_string = isinstance(self.remote_field.model, str)\n        if rel_is_string or not self.requires_unique_target:\n            return []\n\n        try:\n            self.foreign_related_fields\n        except exceptions.FieldDoesNotExist:\n            return []\n\n        if not self.foreign_related_fields:\n            return []\n\n        unique_foreign_fields = {\n            frozenset([f.name])\n            for f in self.remote_field.model._meta.get_fields()\n            if getattr(f, 'unique', False)\n        }\n        unique_foreign_fields.update({\n            frozenset(ut)\n            for ut in self.remote_field.model._meta.unique_together\n        })\n        foreign_fields = {f.name for f in self.foreign_related_fields}\n        has_unique_constraint = any(u <= foreign_fields for u in unique_foreign_fields)\n\n        if not has_unique_constraint and len(self.foreign_related_fields) > 1:\n            field_combination = ', '.join(\n                \"'%s'\" % rel_field.name for rel_field in self.foreign_related_fields\n            )\n            model_name = self.remote_field.model.__name__\n            return [\n                checks.Error(\n                    \"No subset of the fields %s on model '%s' is unique.\"\n                    % (field_combination, model_name),\n                    hint=(\n                        \"Add unique=True on any of those fields or add at \"\n                        \"least a subset of them to a unique_together constraint.\"\n                    ),\n                    obj=self,\n                    id='fields.E310',\n                )\n            ]\n        elif not has_unique_constraint:\n            field_name = self.foreign_related_fields[0].name\n            model_name = self.remote_field.model.__name__\n            return [\n                checks.Error(\n                    \"'%s.%s' must set unique=True because it is referenced by \"\n                    \"a foreign key.\" % (model_name, field_name),\n                    obj=self,\n                    id='fields.E311',\n                )\n            ]\n        else:\n            return []"
      },
      {
        "file": "django/db/models/fields/related.py",
        "type": "function",
        "name": "_check_unique_target",
        "class_name": "ForeignObject",
        "code": "def _check_unique_target(self):\n        rel_is_string = isinstance(self.remote_field.model, str)\n        if rel_is_string or not self.requires_unique_target:\n            return []\n\n        try:\n            self.foreign_related_fields\n        except exceptions.FieldDoesNotExist:\n            return []\n\n        if not self.foreign_related_fields:\n            return []\n\n        unique_foreign_fields = {\n            frozenset([f.name])\n            for f in self.remote_field.model._meta.get_fields()\n            if getattr(f, 'unique', False)\n        }\n        unique_foreign_fields.update({\n            frozenset(ut)\n            for ut in self.remote_field.model._meta.unique_together\n        })\n        foreign_fields = {f.name for f in self.foreign_related_fields}\n        has_unique_constraint = any(u <= foreign_fields for u in unique_foreign_fields)\n\n        if not has_unique_constraint and len(self.foreign_related_fields) > 1:\n            field_combination = ', '.join(\n                \"'%s'\" % rel_field.name for rel_field in self.foreign_related_fields\n            )\n            model_name = self.remote_field.model.__name__\n            return [\n                checks.Error(\n                    \"No subset of the fields %s on model '%s' is unique.\"\n                    % (field_combination, model_name),\n                    hint=(\n                        \"Add unique=True on any of those fields or add at \"\n                        \"least a subset of them to a unique_together constraint.\"\n                    ),\n                    obj=self,\n                    id='fields.E310',\n                )\n            ]\n        elif not has_unique_constraint:\n            field_name = self.foreign_related_fields[0].name\n            model_name = self.remote_field.model.__name__\n            return [\n                checks.Error(\n                    \"'%s.%s' must set unique=True because it is referenced by \"\n                    \"a foreign key.\" % (model_name, field_name),\n                    obj=self,\n                    id='fields.E311',\n                )\n            ]\n        else:\n            return []"
      },
      {
        "file": "django/db/models/fields/related.py",
        "type": "function",
        "name": "_check_unique_target",
        "class_name": "ForeignObject",
        "code": "def _check_unique_target(self):\n        rel_is_string = isinstance(self.remote_field.model, str)\n        if rel_is_string or not self.requires_unique_target:\n            return []\n\n        try:\n            self.foreign_related_fields\n        except exceptions.FieldDoesNotExist:\n            return []\n\n        if not self.foreign_related_fields:\n            return []\n\n        unique_foreign_fields = {\n            frozenset([f.name])\n            for f in self.remote_field.model._meta.get_fields()\n            if getattr(f, 'unique', False)\n        }\n        unique_foreign_fields.update({\n            frozenset(ut)\n            for ut in self.remote_field.model._meta.unique_together\n        })\n        foreign_fields = {f.name for f in self.foreign_related_fields}\n        has_unique_constraint = any(u <= foreign_fields for u in unique_foreign_fields)\n\n        if not has_unique_constraint and len(self.foreign_related_fields) > 1:\n            field_combination = ', '.join(\n                \"'%s'\" % rel_field.name for rel_field in self.foreign_related_fields\n            )\n            model_name = self.remote_field.model.__name__\n            return [\n                checks.Error(\n                    \"No subset of the fields %s on model '%s' is unique.\"\n                    % (field_combination, model_name),\n                    hint=(\n                        \"Add unique=True on any of those fields or add at \"\n                        \"least a subset of them to a unique_together constraint.\"\n                    ),\n                    obj=self,\n                    id='fields.E310',\n                )\n            ]\n        elif not has_unique_constraint:\n            field_name = self.foreign_related_fields[0].name\n            model_name = self.remote_field.model.__name__\n            return [\n                checks.Error(\n                    \"'%s.%s' must set unique=True because it is referenced by \"\n                    \"a foreign key.\" % (model_name, field_name),\n                    obj=self,\n                    id='fields.E311',\n                )\n            ]\n        else:\n            return []"
      }
    ]
  },
  "Justification": "Candidate E is the most helpful because it addresses a similar issue regarding UniqueConstraints in Django models, specifically how the validation checks for uniqueness are handled. This directly relates to the CURRENT bug which also concerns the functionality of UniqueConstraints during the migration process. The insights from Candidate E about how the migration fails due to a lack of proper checks for UniqueConstraints without conditions will be essential in debugging the CURRENT bug's migration issue regarding field existence. Additionally, both involve constraints in database models and how Django interprets them during migrations, making the report particularly relevant to the CURRENT problem.",
  "instance_id": "django__django-12856",
  "repo": "django/django",
  "created_at": "2020-05-04T21:29:23Z",
  "problem_statement": "Add check for fields of UniqueConstraints.\nDescription\n\t \n\t\t(last modified by Marnanel Thurman)\n\t \nWhen a model gains a UniqueConstraint, makemigrations doesn't check that the fields named therein actually exist.\nThis is in contrast to the older unique_together syntax, which raises models.E012 if the fields don't exist.\nIn the attached demonstration, you'll need to uncomment \"with_unique_together\" in settings.py in order to show that unique_together raises E012.\n",
  "patch": "diff --git a/django/db/models/base.py b/django/db/models/base.py\n--- a/django/db/models/base.py\n+++ b/django/db/models/base.py\n@@ -1926,6 +1926,12 @@ def _check_constraints(cls, databases):\n                         id='models.W038',\n                     )\n                 )\n+            fields = (\n+                field\n+                for constraint in cls._meta.constraints if isinstance(constraint, UniqueConstraint)\n+                for field in constraint.fields\n+            )\n+            errors.extend(cls._check_local_fields(fields, 'constraints'))\n         return errors\n \n \n"
}