{
  "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",
  "similar_bug_items": [
    {
      "pr_number": 4655,
      "pr_title": "Fixed #24757 -- Recreated MySQL index when needed during unique_together removal",
      "pr_body": "",
      "issue_id": 24757,
      "issue_title": "Removing unique_together constraint make Django 1.8 migration fail with MySQL",
      "issue_body": "",
      "issue_closed_at": "2015-05-15T10:06:22",
      "base_commit": "0eaef8e527af556c0c517a64035929404cf94a39",
      "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        Deals with a model changing its unique_together.\n        Note: The input unique_togethers must be doubly-nested, not the single-\n        nested [\"foo\", \"bar\"] format.\n        \"\"\"\n        olds = set(tuple(fields) for fields in old_unique_together)\n        news = set(tuple(fields) for fields in new_unique_together)\n        # Deleted uniques\n        for fields in olds.difference(news):\n            columns = [model._meta.get_field(field).column for field in fields]\n            constraint_names = self._constraint_names(model, columns, unique=True)\n            if len(constraint_names) != 1:\n                raise ValueError(\"Found wrong number (%s) of constraints for %s(%s)\" % (\n                    len(constraint_names),\n                    model._meta.db_table,\n                    \", \".join(columns),\n                ))\n            self.execute(self._delete_constraint_sql(self.sql_delete_unique, model, constraint_names[0]))\n        # Created uniques\n        for fields in news.difference(olds):\n            columns = [model._meta.get_field(field).column for field in fields]\n            self.execute(self._create_unique_sql(model, columns))"
        },
        {
          "file": "django/db/backends/base/schema.py",
          "type": "function",
          "name": "alter_index_together",
          "class_name": "BaseDatabaseSchemaEditor",
          "code": "def alter_index_together(self, model, old_index_together, new_index_together):\n        \"\"\"\n        Deals with a model changing its index_together.\n        Note: The input index_togethers must be doubly-nested, not the single-\n        nested [\"foo\", \"bar\"] format.\n        \"\"\"\n        olds = set(tuple(fields) for fields in old_index_together)\n        news = set(tuple(fields) for fields in new_index_together)\n        # Deleted indexes\n        for fields in olds.difference(news):\n            columns = [model._meta.get_field(field).column for field in fields]\n            constraint_names = self._constraint_names(model, list(columns), index=True)\n            if len(constraint_names) != 1:\n                raise ValueError(\"Found wrong number (%s) of constraints for %s(%s)\" % (\n                    len(constraint_names),\n                    model._meta.db_table,\n                    \", \".join(columns),\n                ))\n            self.execute(self._delete_constraint_sql(self.sql_delete_index, model, constraint_names[0]))\n        # Created indexes\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_index_sql(model, fields, suffix=\"_idx\"))"
        },
        {
          "file": "django/db/backends/mysql/schema.py",
          "type": "function",
          "name": "_model_indexes_sql",
          "class_name": "DatabaseSchemaEditor",
          "code": "def _model_indexes_sql(self, model):\n        storage = self.connection.introspection.get_storage_engine(\n            self.connection.cursor(), model._meta.db_table\n        )\n        if storage == \"InnoDB\":\n            for field in model._meta.local_fields:\n                if field.db_index and not field.unique and field.get_internal_type() == \"ForeignKey\":\n                    # Temporary setting db_index to False (in memory) to disable\n                    # index creation for FKs (index automatically created by MySQL)\n                    field.db_index = False\n        return super(DatabaseSchemaEditor, self)._model_indexes_sql(model)"
        }
      ]
    },
    {
      "pr_number": 10199,
      "pr_title": "Fixed #26352 -- Made system check allow ManyToManyField to target the same model if through_fields differs.",
      "pr_body": "See https://code.djangoproject.com/ticket/26352",
      "issue_id": 26352,
      "issue_title": "models.E003 check incorrectly prevents duplicate ManyToMany through-self that differ by through_fields",
      "issue_body": "",
      "issue_closed_at": "2018-08-22T11:28:35",
      "base_commit": "f2d5dafec93e6b3100f004c559ebe21e2b783ae7",
      "changes": [
        {
          "file": "django/db/models/base.py",
          "type": "function",
          "name": "_check_m2m_through_same_relationship",
          "class_name": "Model",
          "code": "def _check_m2m_through_same_relationship(cls):\n        \"\"\" Check if no relationship model is used by more than one m2m field.\n        \"\"\"\n\n        errors = []\n        seen_intermediary_signatures = []\n\n        fields = cls._meta.local_many_to_many\n\n        # Skip when the target model wasn't found.\n        fields = (f for f in fields if isinstance(f.remote_field.model, ModelBase))\n\n        # Skip when the relationship model wasn't found.\n        fields = (f for f in fields if isinstance(f.remote_field.through, ModelBase))\n\n        for f in fields:\n            signature = (f.remote_field.model, cls, f.remote_field.through)\n            if signature in seen_intermediary_signatures:\n                errors.append(\n                    checks.Error(\n                        \"The model has two many-to-many relations through \"\n                        \"the intermediate model '%s'.\" % f.remote_field.through._meta.label,\n                        obj=cls,\n                        id='models.E003',\n                    )\n                )\n            else:\n                seen_intermediary_signatures.append(signature)\n        return errors"
        }
      ]
    },
    {
      "pr_number": 4045,
      "pr_title": "Fixed #15321 -- Honored ancestors unique checks.",
      "pr_body": "",
      "issue_id": 15321,
      "issue_title": "Models._get_unique_checks() does not return grandparents' unique checks",
      "issue_body": "",
      "issue_closed_at": "2015-02-03T17:55:33",
      "base_commit": "1fc458b5a3ec002f51bcac58f09ec77493d8c8c0",
      "changes": [
        {
          "file": "django/db/models/base.py",
          "type": "function",
          "name": "_get_unique_checks",
          "class_name": "Model",
          "code": "def _get_unique_checks(self, exclude=None):\n        \"\"\"\n        Gather a list of checks to perform. Since validate_unique could be\n        called from a ModelForm, some fields may have been excluded; we can't\n        perform a unique check on a model that is missing fields involved\n        in that check.\n        Fields that did not validate should also be excluded, but they need\n        to be passed in via the exclude argument.\n        \"\"\"\n        if exclude is None:\n            exclude = []\n        unique_checks = []\n\n        unique_togethers = [(self.__class__, self._meta.unique_together)]\n        for parent_class in self._meta.parents.keys():\n            if parent_class._meta.unique_together:\n                unique_togethers.append((parent_class, parent_class._meta.unique_together))\n\n        for model_class, unique_together in unique_togethers:\n            for check in unique_together:\n                for name in check:\n                    # If this is an excluded field, don't add this check.\n                    if name in exclude:\n                        break\n                else:\n                    unique_checks.append((model_class, tuple(check)))\n\n        # These are checks for the unique_for_<date/year/month>.\n        date_checks = []\n\n        # Gather a list of checks for fields declared as unique and add them to\n        # the list of checks.\n\n        fields_with_class = [(self.__class__, self._meta.local_fields)]\n        for parent_class in self._meta.parents.keys():\n            fields_with_class.append((parent_class, parent_class._meta.local_fields))\n\n        for model_class, fields in fields_with_class:\n            for f in fields:\n                name = f.name\n                if name in exclude:\n                    continue\n                if f.unique:\n                    unique_checks.append((model_class, (name,)))\n                if f.unique_for_date and f.unique_for_date not in exclude:\n                    date_checks.append((model_class, 'date', name, f.unique_for_date))\n                if f.unique_for_year and f.unique_for_year not in exclude:\n                    date_checks.append((model_class, 'year', name, f.unique_for_year))\n                if f.unique_for_month and f.unique_for_month not in exclude:\n                    date_checks.append((model_class, 'month', name, f.unique_for_month))\n        return unique_checks, date_checks"
        },
        {
          "file": "django/db/models/base.py",
          "type": "function",
          "name": "_get_unique_checks",
          "class_name": "Model",
          "code": "def _get_unique_checks(self, exclude=None):\n        \"\"\"\n        Gather a list of checks to perform. Since validate_unique could be\n        called from a ModelForm, some fields may have been excluded; we can't\n        perform a unique check on a model that is missing fields involved\n        in that check.\n        Fields that did not validate should also be excluded, but they need\n        to be passed in via the exclude argument.\n        \"\"\"\n        if exclude is None:\n            exclude = []\n        unique_checks = []\n\n        unique_togethers = [(self.__class__, self._meta.unique_together)]\n        for parent_class in self._meta.parents.keys():\n            if parent_class._meta.unique_together:\n                unique_togethers.append((parent_class, parent_class._meta.unique_together))\n\n        for model_class, unique_together in unique_togethers:\n            for check in unique_together:\n                for name in check:\n                    # If this is an excluded field, don't add this check.\n                    if name in exclude:\n                        break\n                else:\n                    unique_checks.append((model_class, tuple(check)))\n\n        # These are checks for the unique_for_<date/year/month>.\n        date_checks = []\n\n        # Gather a list of checks for fields declared as unique and add them to\n        # the list of checks.\n\n        fields_with_class = [(self.__class__, self._meta.local_fields)]\n        for parent_class in self._meta.parents.keys():\n            fields_with_class.append((parent_class, parent_class._meta.local_fields))\n\n        for model_class, fields in fields_with_class:\n            for f in fields:\n                name = f.name\n                if name in exclude:\n                    continue\n                if f.unique:\n                    unique_checks.append((model_class, (name,)))\n                if f.unique_for_date and f.unique_for_date not in exclude:\n                    date_checks.append((model_class, 'date', name, f.unique_for_date))\n                if f.unique_for_year and f.unique_for_year not in exclude:\n                    date_checks.append((model_class, 'year', name, f.unique_for_year))\n                if f.unique_for_month and f.unique_for_month not in exclude:\n                    date_checks.append((model_class, 'month', name, f.unique_for_month))\n        return unique_checks, date_checks"
        }
      ]
    },
    {
      "pr_number": 4352,
      "pr_title": "Fixed #24508 -- Made annotations reflective",
      "pr_body": "We should backport this to 1.8 to be consistent with `filter(a=2+F())`.\n",
      "issue_id": 24508,
      "issue_title": "F() object operations do not correcly reflect with annotate",
      "issue_body": "",
      "issue_closed_at": "2015-03-22T01:34:12",
      "base_commit": "a6bada1ee0c3756e4b8d6bd4b4346dd5235c78ce",
      "changes": [
        {
          "file": "django/db/models/expressions.py",
          "type": "function",
          "name": "_resolve_output_field",
          "class_name": "BaseExpression",
          "code": "def _resolve_output_field(self):\n        \"\"\"\n        Attempts to infer the output type of the expression. If the output\n        fields of all source fields match then we can simply infer the same\n        type here.\n        \"\"\"\n        if self._output_field is None:\n            sources = self.get_source_fields()\n            num_sources = len(sources)\n            if num_sources == 0:\n                self._output_field = None\n            else:\n                self._output_field = sources[0]\n                for source in sources:\n                    if source is not None and not isinstance(self._output_field, source.__class__):\n                        raise FieldError(\n                            \"Expression contains mixed types. You must set output_field\")"
        },
        {
          "file": "django/db/models/expressions.py",
          "type": "function",
          "name": "_resolve_output_field",
          "class_name": "BaseExpression",
          "code": "def _resolve_output_field(self):\n        \"\"\"\n        Attempts to infer the output type of the expression. If the output\n        fields of all source fields match then we can simply infer the same\n        type here.\n        \"\"\"\n        if self._output_field is None:\n            sources = self.get_source_fields()\n            num_sources = len(sources)\n            if num_sources == 0:\n                self._output_field = None\n            else:\n                self._output_field = sources[0]\n                for source in sources:\n                    if source is not None and not isinstance(self._output_field, source.__class__):\n                        raise FieldError(\n                            \"Expression contains mixed types. You must set output_field\")"
        }
      ]
    },
    {
      "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": "",
      "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 []"
        }
      ]
    }
  ]
}