{
  "id": "django__django-11905",
  "question": "Prevent using __isnull lookup with non-boolean value.\nDescription\n\t \n\t\t(last modified by Mariusz Felisiak)\n\t \n__isnull should not allow for non-boolean values. Using truthy/falsey doesn't promote INNER JOIN to an OUTER JOIN but works fine for a simple queries. Using non-boolean values is ​undocumented and untested. IMO we should raise an error for non-boolean values to avoid confusion and for consistency.\n",
  "answer": "{\"task_id\": \"67b62fa4-5435-482f-af53-25e8b64e2f6f\", \"workflow_run_id\": \"3981866d-1a04-40fd-968f-7f4fdd67bad7\", \"data\": {\"id\": \"3981866d-1a04-40fd-968f-7f4fdd67bad7\", \"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/db/models/lookups.py b/django/db/models/lookups.py\\\\nindex af5b8cd..b32e4d1 100644\\\\n--- a/django/db/models/lookups.py\\\\n+++ b/django/db/models/lookups.py\\\\n@@ -47,6 +47,10 @@ class IsNull(models.Lookup):\\\\n \\\\n     def get_prep_lookup(self):\\\\n         # Ensure the rhs value is boolean.\\\\n+        if not isinstance(self.rhs, bool):\\\\n+            raise ValueError('Invalid value for __isnull lookup. Only boolean values are allowed.')\\\\n+\\\\n         return super().get_prep_lookup()\\\\n \\\\n\\\",\\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 enforcing that only boolean values are allowed for __isnull lookups. It raises a clear error message, which aligns with best practices for input validation. The risk of introducing new bugs is minimal as it only adds a type check.\\\",\\n      \\\"recommendation\\\": \\\"Highly recommended\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 3,\\n      \\\"patch\\\": \\\"diff --git a/django/db/models/lookups.py b/django/db/models/lookups.py\\\\nindex af5b8cd..b32e4d1 100644\\\\n--- a/django/db/models/lookups.py\\\\n+++ b/django/db/models/lookups.py\\\\n@@ -47,6 +47,11 @@ class IsNull(models.Lookup):\\\\n \\\\n     def get_prep_lookup(self):\\\\n         # Ensure the rhs value is explicitly True or False.\\\\n+        if self.rhs not in [True, False]:\\\\n+            raise ValueError('Invalid value for __isnull lookup. Only True or False are allowed.')\\\\n+\\\\n         return super().get_prep_lookup()\\\\n \\\\n\\\",\\n      \\\"rank\\\": 2,\\n      \\\"score\\\": 0.9,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.85,\\n        \\\"code_quality\\\": 0.9,\\n        \\\"risk_level\\\": 0.1,\\n        \\\"best_practices\\\": 0.85,\\n        \\\"test_coverage\\\": 0.8\\n      },\\n      \\\"reasoning\\\": \\\"This patch enforces stricter checking by allowing only explicit True or False values. While this is a valid approach, it might be too restrictive compared to the typical Pythonic way of handling booleans. The risk is low, but it could potentially break existing code that relies on truthy/falsey values.\\\",\\n      \\\"recommendation\\\": \\\"Recommended with caution\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 2,\\n      \\\"patch\\\": \\\"diff --git a/django/db/models/lookups.py b/django/db/models/lookups.py\\\\nindex af5b8cd..b32e4d1 100644\\\\n--- a/django/db/models/lookups.py\\\\n+++ b/django/db/models/lookups.py\\\\n@@ -47,6 +47,12 @@ class IsNull(models.Lookup):\\\\n \\\\n     def get_prep_lookup(self):\\\\n         # Ensure the rhs value is boolean.\\\\n+        if not isinstance(self.rhs, (bool, type(None))):\\\\n+            raise ValueError('`rhs` should be a boolean or None.')\\\\n+\\\\n         return super().get_prep_lookup()\\\\n \\\\n\\\",\\n      \\\"rank\\\": 3,\\n      \\\"score\\\": 0.85,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.8,\\n        \\\"code_quality\\\": 0.85,\\n        \\\"risk_level\\\": 0.15,\\n        \\\"best_practices\\\": 0.8,\\n        \\\"test_coverage\\\": 0.75\\n      },\\n      \\\"reasoning\\\": \\\"This patch allows None as a valid value, which is not typical for __isnull lookups and could introduce ambiguity. While it provides flexibility, it deviates from the standard expectation of boolean-only values, potentially leading to confusion.\\\",\\n      \\\"recommendation\\\": \\\"Not recommended due to potential ambiguity\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 4,\\n      \\\"patch\\\": \\\"diff --git a/tests/field_lookups/tests.py b/tests/field_lookups/tests.py\\\\nindex d2f7e57..7aa6c3d 100644\\\\n--- a/tests/field_lookups/tests.py\\\\n+++ b/tests/field_lookups/tests.py\\\\n@@ -202,6 +202,19 @@ class FieldLookupTests(TestCase):\\\\n         )\\\\n \\\\n+    def test_isnull_non_boolean_value(self):\\\\n+        with self.assertRaises(ValueError) as cm:\\\\n+            SomeModel.objects.filter(name__isnull='non_boolean_value')\\\\n+        self.assertEqual(\\\\n+            str(cm.exception),\\\\n+            'Invalid value for __isnull lookup. Only boolean values are allowed.'\\\\n+        )\\\\n+\\\\n+        with self.assertRaises(ValueError) as cm:\\\\n+            SomeModel.objects.filter(name__isnull=1)\\\\n+        self.assertEqual(\\\\n+            str(cm.exception),\\\\n+            'Invalid value for __isnull lookup. Only boolean values are allowed.'\\\\n+        )\\\\n+\\\\n\\\",\\n      \\\"rank\\\": 4,\\n      \\\"score\\\": 0.8,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.75,\\n        \\\"code_quality\\\": 0.8,\\n        \\\"risk_level\\\": 0.05,\\n        \\\"best_practices\\\": 0.75,\\n        \\\"test_coverage\\\": 0.9\\n      },\\n      \\\"reasoning\\\": \\\"This patch adds a test case to ensure non-boolean values raise an error. While it is useful for validation, it does not address the core issue directly and should be considered supplementary to the other patches.\\\",\\n      \\\"recommendation\\\": \\\"Useful for validation but not a standalone fix\\\"\\n    }\\n  ],\\n  \\\"evaluation_summary\\\": \\\"Patch 1 is the best choice as it directly addresses the issue by enforcing boolean-only values for __isnull lookups, aligning with best practices and minimizing risk. Patch 3 is a close second, offering stricter enforcement but potentially being too restrictive. Patch 2 introduces ambiguity by allowing None, which is not recommended. Patch 4 provides valuable test coverage but does not fix the issue itself.\\\"\\n}\\n```\", \"generated_tests\": \"```json\\n{\\n  \\\"reproduction_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_reproduce_original_issue\\\",\\n      \\\"test_code\\\": \\\"def test_reproduce_original_issue():\\\\n    # Test creating a filter using non-boolean values with __isnull\\\\n    with self.assertRaises(ValueError) as cm:\\\\n        SomeModel.objects.filter(name__isnull='non_boolean_value')\\\\n    message = str(cm.exception)\\\\n    assert message == 'Invalid value for __isnull lookup. Only boolean values are allowed.'\\\\n\\\\n    with self.assertRaises(ValueError) as cm:\\\\n        SomeModel.objects.filter(name__isnull=1)\\\\n    message = str(cm.exception)\\\\n    assert message == 'Invalid value for __isnull lookup. Only boolean values are allowed.'\\\",\\n      \\\"description\\\": \\\"This test reproduces the original issue by creating filters using non-boolean values with __isnull\\\",\\n      \\\"expected_behavior\\\": \\\"The test should fail with a ValueError before applying the patches, and should pass after applying the patches.\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_edge_cases\\\",\\n      \\\"test_code\\\": \\\"def test_edge_cases():\\\\n    # Test creating a filter using boolean values with __isnull\\\\n    SomeModel.objects.filter(name__isnull=True)\\\\n    SomeModel.objects.filter(name__isnull=False)\\\\n\\\\n    # Test creating a filter using None with __isnull\\\\n    SomeModel.objects.filter(name__isnull=None)\\\\n\\\\n    # Test creating a filter using wrong boolean values with __isnull\\\\n    with self.assertRaises(ValueError) as cm:\\\\n        SomeModel.objects.filter(name__isnull='True')\\\\n    message = str(cm.exception)\\\\n    assert message == '`rhs` should be a boolean or None.'\\\",\\n      \\\"description\\\": \\\"This test covers edge cases by using boolean values, None, and wrong boolean values with __isnull\\\",\\n      \\\"expected_behavior\\\": \\\"All filters should work correctly for valid boolean values and None, and raise a ValueError for wrong boolean values.\\\"\\n    }\\n  ],\\n  \\\"validation_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation_p1\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation_p1():\\\\n    # Test applying patch 1\\\\n    # Test creating a filter using non-boolean values with __isnull\\\\n    with self.assertRaises(ValueError) as cm:\\\\n        SomeModel.objects.filter(name__isnull='non_boolean_value')\\\\n    message = str(cm.exception)\\\\n    assert message == 'Invalid value for __isnull lookup. Only boolean values are allowed.'\\\",\\n      \\\"description\\\": \\\"This test validates that patch 1 correctly raises a ValueError for non-boolean values with __isnull\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation_p2\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation_p2():\\\\n    # Test applying patch 2\\\\n    # Test creating a filter using None with __isnull\\\\n    SomeModel.objects.filter(name__isnull=None)\\\",\\n      \\\"description\\\": \\\"This test validates that patch 2 allows for None as a valid value with __isnull\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation_p3\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation_p3():\\\\n    # Test applying patch 3\\\\n    # Test creating a filter using explicitly True or False with __isnull\\\\n    SomeModel.objects.filter(name__isnull=True)\\\\n    SomeModel.objects.filter(name__isnull=False)\\\",\\n      \\\"description\\\": \\\"This test validates that patch 3 enforces using explicitly True or False with __isnull\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation_p4\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation_p4():\\\\n    # Test applying patch 4\\\\n    # Test creating a filter using non-boolean values with __isnull\\\\n    with self.assertRaises(ValueError) as cm:\\\\n        SomeModel.objects.filter(name__isnull='non_boolean_value')\\\\n    message = str(cm.exception)\\\\n    assert message == 'Invalid value for __isnull lookup. Only boolean values are allowed.'\\\",\\n      \\\"description\\\": \\\"This test validates that patch 4 correctly raises a ValueError for non-boolean values with __isnull\\\"\\n    }\\n  ],\\n  \\\"test_summary\\\": \\\"Comprehensive test cases have been generated to reproduce the original issue, test edge cases related to it, and validate the patches provided. The patches 1, 2, and 3 aim to address the original issue by enforcing boolean values or None for __isnull, while patch 4 provides a test case to ensure the error is raised for non-boolean values.\\\"\\n}\\n```\"}, \"error\": \"\", \"elapsed_time\": 391.831552, \"total_tokens\": 22915, \"total_steps\": 9, \"created_at\": 1753287675, \"finished_at\": 1753288067}}"
}