{
  "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": "Migrations containing\nmigrations.AlterUniqueTogether(name='xxx', unique_together=set([]),)\nmay fail on Django 1.8 but not on Django 1.7 if using the backend mysql.\nGiven the database engine is mysql and the following\nmodels.py\n:\nfrom django.db import models\n\n\nclass MyModelA(models.Model):\n    s = models.CharField(max_length=120)\n\n\nclass MyModelB(models.Model):\n    a = models.ForeignKey(MyModelA)\n    b = models.IntegerField()\nAnd the 2 following migrations:\n# -*- coding: utf-8 -*-\nfrom __future__ import unicode_literals\n\nfrom django.db import models, migrations\n\n\nclass Migration(migrations.Migration):\n\n    dependencies = [\n    ]\n\n    operations = [\n        migrations.CreateModel(\n            name='MyModelA',\n            fields=[\n                ('id', models.AutoField(serialize=False, auto_created=True, verbose_name='ID', primary_key=True)),\n                ('s', models.CharField(max_length=120)),\n            ],\n        ),\n        migrations.CreateModel(\n            name='MyModelB',\n            fields=[\n                ('id', models.AutoField(serialize=False, auto_created=True, verbose_name='ID', primary_key=True)),\n                ('b', models.IntegerField()),\n                ('a', models.ForeignKey(to='mysite.MyModelA')),\n            ],\n        ),\n        migrations.AlterUniqueTogether(\n            name='mymodelb',\n            unique_together=set([('a', 'b')]),\n        ),\n    ]\n# -*- coding: utf-8 -*-\nfrom __future__ import unicode_literals\n\nfrom django.db import models, migrations\n\n\nclass Migration(migrations.Migration):\n\n    dependencies = [\n        ('mysite', '0001_initial'),\n    ]\n\n    operations = [\n        migrations.AlterUniqueTogether(\n            name='mymodelb',\n            unique_together=set([]),\n        ),\n    ]\nRunning\n./manage.py migrate\nworks fine on Django 1.7, but on Django 1.8 I get the following error:\n$ ./manage.py migrate                \nOperations to perform:\n  Synchronize unmigrated apps: staticfiles, messages\n  Apply all migrations: contenttypes, auth, mysite, sessions, admin\nSynchronizing apps without migrations:\n  Creating tables...\n    Running deferred SQL...\n  Installing custom SQL...\nRunning migrations:\n  Rendering model states... DONE\n  [..]\n  Applying mysite.0001_initial... OK\n  Applying mysite.0002_auto...Traceback (most recent call last):\n  [..]\ndjango.db.utils.OperationalError: (1553, \"Cannot drop index 'mysite_mymodelb_a_id_b53c781c93aab9a_uniq': needed in a foreign key constraint\")\nThe same migrations work on psql, I did not have the chance to test it against sqlite.\nInspecting the generated sql I found Django 1.7 add an index in addition to the unique constraint and Django 1.8 does not.\nOn Django 1.7\nBEGIN;\nCREATE TABLE `mysite_mymodela` (`id` integer AUTO_INCREMENT NOT NULL PRIMARY KEY, `s` varchar(120) NOT NULL);\nCREATE TABLE `mysite_mymodelb` (`id` integer AUTO_INCREMENT NOT NULL PRIMARY KEY, `b` integer NOT NULL, `a_id` integer NOT NULL);\nALTER TABLE `mysite_mymodelb` ADD CONSTRAINT `mysite_mymodelb_a_id_48275010b87402ed_uniq` UNIQUE (`a_id`, `b`);\nALTER TABLE `mysite_mymodelb` ADD CONSTRAINT `mysite_mymodelb_a_id_4981d3920f0fef1e_fk_mysite_mymodela_id` FOREIGN KEY (`a_id`) REFERENCES `mysite_mymodela` (`id`);\nCREATE INDEX `mysite_mymodelb_e2b453f4` ON `mysite_mymodelb` (`a_id`);\n\nCOMMIT;\nNote the\nCREATE INDEX `mysite_mymodelb_e2b453f4` ON `mysite_mymodelb` (`a_id`);\n.\nAnd on Django 1.8\nBEGIN;\nCREATE TABLE `mysite_mymodela` (`id` integer AUTO_INCREMENT NOT NULL PRIMARY KEY, `s` varchar(120) NOT NULL);\nCREATE TABLE `mysite_mymodelb` (`id` integer AUTO_INCREMENT NOT NULL PRIMARY KEY, `b` integer NOT NULL, `a_id` integer NOT NULL);\nALTER TABLE `mysite_mymodelb` ADD CONSTRAINT `mysite_mymodelb_a_id_784bb266d0e363ab_uniq` UNIQUE (`a_id`, `b`);\nALTER TABLE `mysite_mymodelb` ADD CONSTRAINT `mysite_mymodelb_a_id_50892c78e36678b8_fk_mysite_mymodela_id` FOREIGN KEY (`a_id`) REFERENCES `mysite_mymodela` (`id`);\n\nCOMMIT;\nI joined the full traceback.",
      "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": "Let's say we're building a Twitter-style friendship model, where a user can follow other users, and can have users that follow them.\nA sensible way to model that might look like this:\nclass\nUser\n(\nmodels\n.\nModel\n):\nfriends\n=\nmodels\n.\nManyToManyField\n(\n'self'\n,\nthrough\n=\n'Followship'\n,\nsymmetrical\n=\nFalse\n,\nthrough_fields\n=\n(\n'user'\n,\n'target'\n),\nrelated_name\n=\n'+'\n)\nfollowers\n=\nmodels\n.\nManyToManyField\n(\n'self'\n,\nthrough\n=\n'Followship'\n,\nsymmetrical\n=\nFalse\n,\nthrough_fields\n=\n(\n'target'\n,\n'user'\n),\nrelated_name\n=\n'+'\n)\nclass\nFollowship\n(\nmodels\n.\nModel\n):\nuser\n=\nmodels\n.\nForeignKey\n(\nUser\n,\nmodels\n.\nCASCADE\n,\nrelated_name\n=\n'+'\n)\ntarget\n=\nmodels\n.\nForeignKey\n(\nUser\n,\nmodels\n.\nCASCADE\n,\nrelated_name\n=\n'+'\n)\nHere we have an intermediary table called \"Followship\", which relates a user to the target user that they are following.\nWe also have two helpful convenience ManyToMany fields defined on the user:\nuser.friends.all()\nreturns all other users that the user is following, while\nuser.followers.all()\nreturns the users that are following our current user.\nThe above models should work fine... but they don't, because Django's model checking framework throws the following error:\nusers.User: (models.E003) The model has two many-to-many relations through the intermediate model 'users.Followship'.\nI don't think this warning is warranted in this case, because the\nthrough_fields\narguments provided to the friends/followers ManyToManyFields mean that the relationships defined here make sense and the models should work correctly.\nI've tested this theory by disabling the warning entirely using an over-ride class method on user, like this:\nclass\nUser\n(\nmodels\n.\nModel\n):\nfriends\n=\nmodels\n.\nManyToManyField\n(\n'self'\n,\nthrough\n=\n'Followship'\n,\nsymmetrical\n=\nFalse\n,\nthrough_fields\n=\n(\n'user'\n,\n'target'\n),\nrelated_name\n=\n'+'\n)\nfollowers\n=\nmodels\n.\nManyToManyField\n(\n'self'\n,\nthrough\n=\n'Followship'\n,\nsymmetrical\n=\nFalse\n,\nthrough_fields\n=\n(\n'target'\n,\n'user'\n),\nrelated_name\n=\n'+'\n)\n@classmethod\ndef\n_check_m2m_through_same_relationship\n(\ncls\n):\n# Disable models.E003 check for this model\nreturn\n[]\nThis does the trick: the check is suppressed, and the models work as expected.\nI think the model check framework is being overly strict here. I think it should be modified to enable this pattern, provided the through_fields= parameters on the duplicate ManyToMany fields are present and differentiate the fields correctly.",
      "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": "Consider the following models:\nclass\nProject\n(\nmodels\n.\nModel\n):\ntitle\n=\nmodels\n.\nCharField\n(\nmax_length\n=\n255\n)\ninternal_id\n=\nmodels\n.\nCharField\n(\nmax_length\n=\n255\n,\nunique\n=\nTrue\n)\nclass\nSubProject\n(\nProject\n):\n# ... fields of SubProject ...\npass\nclass\nSubSubProject\n(\nSubProject\n):\n# ... fields of SubSubProject ...\npass\nFollowing code works as expected (b.full_clean() throws a ValidationError):\na\n=\nSubProject\n(\ntitle\n=\n'A'\n,\ninternal_id\n=\n1\n)\na\n.\nsave\n()\nb\n=\nSubProject\n(\ntitle\n=\n'B'\n,\ninternal_id\n=\n1\n)\nb\n.\nfull_clean\n()\n# Throws ValidationError\nFollowing code does not work as expected (b.full_clean() does not throw an exception, b.save() throws an\nIntegrityError\n):\na\n=\nSubSubProject\n(\ntitle\n=\n'A'\n,\ninternal_id\n=\n2\n)\na\n.\nsave\n()\nb\n=\nSubSubProject\n(\ntitle\n=\n'B'\n,\ninternal_id\n=\n2\n)\nb\n.\nfull_clean\n()\n# Does not throw any exception!\nb\n.\nsave\n()\n# Throws IntegrityError, of course!",
      "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": "Hello,\nThis ticket is related to the 1.8 new feature of using various F operations within an annotation.\nI have spotted 2 problems so far :\n1) F('field') * 2 apparently isn't the same as 2 * F('field')\nDescription of the problem :\nSomeModel.objects.all().annotate(computed = F('some_field') * 2)\nWorks as expected\n\nSomeModel.objects.all().annotate(computed = 2 * F('some_field'))\nE: django.core.exceptions.FieldError: Expression contains mixed types. You must set output_field\n\nSomeModel.objects.all().annotate(computed = Expression(2 * F('some_field'), output_field = FloatField()))\nE : TypeError: __init__() got multiple values for argument 'output_field'\nThe last exception, about init, is the most problematic as it doesn't state anything about where is the problem (no info in the traceback)\n2) When  a F object is added to None, it causes a conflict\nDescription of the problem :\nSomeModel.objects.all().annotate(computed = F('some_field') + None)\nWorks as expected\n\nSomeModel.objects.all().annotate(computed = None + F('some_field'))\nE: django.core.exceptions.FieldError: Expression contains mixed types. You must set output_field\n\nSomeModel.objects.all().annotate(computed = Expression(None + F('some_field'), output_field = FloatField()))\nE : TypeError: __init__() got multiple values for argument 'output_field'\nYou might wonder why there would be a None in the construction. For me it is because I am building the object in a for, so I have something like this :\nannotation = None\nfor field in fields_to_be_added:\n    annotation += F(field)\nSomeModel.objects.all().annotate(annotation)",
      "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": "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 []"
        }
      ]
    }
  ]
}