{
  "id": "django__django-12856",
  "question": "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",
  "answer": "{\"task_id\": \"323b1aee-0692-460d-acb5-bb23a938ee17\", \"workflow_run_id\": \"1ea16a92-e4b1-4560-9247-e8da5ee355bf\", \"data\": {\"id\": \"1ea16a92-e4b1-4560-9247-e8da5ee355bf\", \"workflow_id\": \"bc5a1a30-852f-4fd7-98f0-8fca2f23c564\", \"status\": \"succeeded\", \"outputs\": {\"ranked_patches\": \"```json\\n{\\n  \\\"ranked_patches\\\": [\\n    {\\n      \\\"patch_id\\\": 1,\\n      \\\"patch\\\": \\\"diff --git a/django/core/checks/model_checks.py b/django/core/checks/model_checks.py\\\\nindex a1b2c3d..e4f5g6h 100644\\\\n--- a/django/core/checks/model_checks.py\\\\n+++ b/django/core/checks/model_checks.py\\\\n@@ -11,6 +11,7 @@\\\\n from django.core.exceptions import FieldDoesNotExist\\\\n from django.core.checks import Error\\\\n \\\\n+E013 = 'models.E013'\\\\n \\\\n def check_model(model):\\\\n     errors = []\\\\n@@ -151,6 +152,22 @@ def check_model(model):\\\\n         for field_name in options.unique_together:\\\\n             errors.append(check_field_name(model, field_name))\\\\n \\\\n+    # Check fields for UniqueConstraints\\\\n+    for constraint in options.constraints:\\\\n+        if isinstance(constraint, UniqueConstraint):\\\\n+            for field_name in constraint.fields:\\\\n+                try:\\\\n+                    model._meta.get_field(field_name)\\\\n+                except FieldDoesNotExist:\\\\n+                    errors.append(\\\\n+                        Error(\\\\n+                            f\\\\\\\"Field '{field_name}' in unique constraint '{constraint.name}' does not exist.\\\\\\\",\\\\n+                            obj=model,\\\\n+                            id=E013,\\\\n+                        )\\\\n+                    )\\\\n+                    \\\\n+\\\\n     return errors\\\\n \\\\n def check_field_name(model, field_name):\\\",\\n      \\\"rank\\\": 1,\\n      \\\"score\\\": 0.95,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.9,\\n        \\\"code_quality\\\": 0.95,\\n        \\\"risk_level\\\": 0.1,\\n        \\\"best_practices\\\": 0.9,\\n        \\\"test_coverage\\\": 0.85\\n      },\\n      \\\"reasoning\\\": \\\"This patch correctly addresses the issue by adding a check for field existence in UniqueConstraints using Django's existing field retrieval method, which is robust and aligns with existing practices. The patch is straightforward, minimizes risk, and integrates well with the existing code structure.\\\",\\n      \\\"recommendation\\\": \\\"Highly recommended\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 3,\\n      \\\"patch\\\": \\\"diff --git a/django/core/checks/model_checks.py b/django/core/checks/model_checks.py\\\\nindex a1b2c3d..e4f5g6h 100644\\\\n--- a/django/core/checks/model_checks.py\\\\n+++ b/django/core/checks/model_checks.py\\\\n@@ -11,6 +11,7 @@ from django.core.exceptions import FieldDoesNotExist\\\\n from django.core.checks import Error\\\\n \\\\n E012 = 'models.E012'\\\\n+E013 = 'models.E013'\\\\n \\\\n def check_model(model):\\\\n     errors = []\\\\n@@ -150,6 +151,23 @@ def check_model(model):\\\\n             )\\\\n         )\\\\n \\\\n+    # Check UniqueConstraints fields\\\\n+    for constraint in model._meta.constraints:\\\\n+        if isinstance(constraint, UniqueConstraint):\\\\n+            missing_fields = [\\\\n+                field_name for field_name in constraint.fields\\\\n+                if not model._meta.get_field(field_name, fail_silently=True)\\\\n+            ]\\\\n+            if missing_fields:\\\\n+                errors.append(\\\\n+                    Error(\\\\n+                        \\\\\\\"Field(s) '%s' in unique constraint '%s' do not exist.\\\\\\\"\\\\n+                        % (', '.join(missing_fields), constraint.name),\\\\n+                        obj=model,\\\\n+                        id=E013,\\\\n+                    )\\\\n+                )\\\\n+    \\\\n     return errors\\\\n \\\\n def check_field_name(model, field_name):\\\",\\n      \\\"rank\\\": 2,\\n      \\\"score\\\": 0.9,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.85,\\n        \\\"code_quality\\\": 0.9,\\n        \\\"risk_level\\\": 0.15,\\n        \\\"best_practices\\\": 0.85,\\n        \\\"test_coverage\\\": 0.8\\n      },\\n      \\\"reasoning\\\": \\\"This patch also addresses the issue effectively by checking for missing fields in UniqueConstraints. It uses a list comprehension to gather missing fields, which is efficient. However, the use of 'fail_silently=True' could potentially mask other issues, slightly increasing the risk.\\\",\\n      \\\"recommendation\\\": \\\"Recommended with caution\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 2,\\n      \\\"patch\\\": \\\"diff --git a/django/core/checks/model_checks.py b/django/core/checks/model_checks.py\\\\nindex a1b2c3d..e4f5g6h 100644\\\\n--- a/django/core/checks/model_checks.py\\\\n+++ b/django/core/checks/model_checks.py\\\\n@@ -10,6 +10,7 @@ from django.core.exceptions import FieldDoesNotExist\\\\n from django.core.checks import Error\\\\n \\\\n E012 = 'models.E012'\\\\n+E013 = 'models.E013'\\\\n \\\\n def check_model(model):\\\\n     errors = []\\\\n@@ -160,6 +161,20 @@ def check_model(model):\\\\n                     obj=model,\\\\n                     id=E012,\\\\n                 )\\\\n+            )\\\\n+\\\\n+    # Check that fields in UniqueConstraints exist in the model\\\\n+    for constraint in model._meta.constraints:\\\\n+        if isinstance(constraint, UniqueConstraint):\\\\n+            for field_name in constraint.fields:\\\\n+                if field_name not in model._meta.fields_map:\\\\n+                    errors.append(\\\\n+                        Error(\\\\n+                            \\\\\\\"Field '%s' specified in unique constraint '%s' does not exist.\\\\\\\"\\\\n+                            % (field_name, constraint.name),\\\\n+                            obj=model,\\\\n+                            id=E013,\\\\n+                        )\\\\n                 )\\\\n \\\\n     return errors\\\\n\\\",\\n      \\\"rank\\\": 3,\\n      \\\"score\\\": 0.85,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.8,\\n        \\\"code_quality\\\": 0.85,\\n        \\\"risk_level\\\": 0.2,\\n        \\\"best_practices\\\": 0.8,\\n        \\\"test_coverage\\\": 0.75\\n      },\\n      \\\"reasoning\\\": \\\"This patch checks for field existence using 'fields_map', which is less direct than using 'get_field'. It may not cover all cases where fields are defined in different ways, increasing the risk of missing some errors. The approach is less aligned with Django's typical field access patterns.\\\",\\n      \\\"recommendation\\\": \\\"Least recommended\\\"\\n    }\\n  ],\\n  \\\"evaluation_summary\\\": \\\"Patch 1 is the most robust and aligns well with Django's existing practices for field checks, making it the top recommendation. Patch 3 is also effective but uses a method that could potentially obscure other issues. Patch 2, while functional, uses a less direct method for checking field existence, which could lead to missed errors, making it the least recommended.\\\"\\n}\\n```\", \"generated_tests\": \"```json\\n{\\n  \\\"reproduction_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_reproduce_issue_missing_field_uniqueconstraint\\\",\\n      \\\"test_code\\\": \\\"def test_reproduce_issue_missing_field_uniqueconstraint():\\\\n    # Test model with a UniqueConstraint containing a missing field\\\\n    class TestModel(models.Model):\\\\n        name = models.CharField(max_length=50)\\\\n        age = models.IntegerField()\\\\n\\\\n        class Meta:\\\\n            constraints = [models.UniqueConstraint(fields=['name', 'email'], name='unique_name_email')]\\\\n    \\\\n    errors = check_model(TestModel)\\\\n    assert any(error.id == 'models.E013' for error in errors)\\\\n\\\",\\n      \\\"description\\\": \\\"This test reproduces the issue by defining a model with a UniqueConstraint containing a missing field\\\",\\n      \\\"expected_behavior\\\": \\\"The check_model function should detect the missing field in the UniqueConstraint and raise an error with id 'models.E013'\\\"\\n    }\\n  ],\\n  \\\"validation_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation_patch1\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation_patch1():\\\\n    # Test validation for Patch 1\\\\n    class TestModel(models.Model):\\\\n        name = models.CharField(max_length=50)\\\\n        age = models.IntegerField()\\\\n\\\\n        class Meta:\\\\n            constraints = [models.UniqueConstraint(fields=['name', 'email'], name='unique_name_email')]\\\\n    \\\\n    errors = check_model(TestModel)\\\\n    assert all(error.id != 'models.E013' for error in errors)\\\\n\\\",\\n      \\\"description\\\": \\\"This test validates Patch 1 which adds field existence check for UniqueConstraint in model_checks.py\\\",\\n      \\\"expected_behavior\\\": \\\"The check_model function should not raise any error with id 'models.E013' after applying Patch 1\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation_patch2\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation_patch2():\\\\n    # Test validation for Patch 2\\\\n    class TestModel(models.Model):\\\\n        name = models.CharField(max_length=50)\\\\n        age = models.IntegerField()\\\\n\\\\n        class Meta:\\\\n            constraints = [models.UniqueConstraint(fields=['name', 'email'], name='unique_name_email')]\\\\n    \\\\n    errors = check_model(TestModel)\\\\n    assert all(error.id != 'models.E013' for error in errors)\\\\n\\\",\\n      \\\"description\\\": \\\"This test validates Patch 2 which adds field existence check for UniqueConstraint in model_checks.py\\\",\\n      \\\"expected_behavior\\\": \\\"The check_model function should not raise any error with id 'models.E013' after applying Patch 2\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation_patch3\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation_patch3():\\\\n    # Test validation for Patch 3\\\\n    class TestModel(models.Model):\\\\n        name = models.CharField(max_length=50)\\\\n        age = models.IntegerField()\\\\n\\\\n        class Meta:\\\\n            constraints = [models.UniqueConstraint(fields=['name', 'email'], name='unique_name_email')]\\\\n    \\\\n    errors = check_model(TestModel)\\\\n    assert all(error.id != 'models.E013' for error in errors)\\\\n\\\",\\n      \\\"description\\\": \\\"This test validates Patch 3 which adds field existence check for UniqueConstraint in model_checks.py\\\",\\n      \\\"expected_behavior\\\": \\\"The check_model function should not raise any error with id 'models.E013' after applying Patch 3\\\"\\n    }\\n  ],\\n  \\\"test_summary\\\": \\\"Comprehensive test cases generated to reproduce the original issue and validate the three proposed patches for adding field existence checks for UniqueConstraints in Django's check_model function.\\\"\\n}\\n```\"}, \"error\": \"\", \"elapsed_time\": 375.299512, \"total_tokens\": 19222, \"total_steps\": 9, \"created_at\": 1753293879, \"finished_at\": 1753294254}}"
}