{
  "id": "django__django-13158",
  "question": "QuerySet.none() on combined queries returns all results.\nDescription\n\t\nI came across this issue on Stack Overflow. I'm not 100% sure it's a bug, but it does seem strange. With this code (excuse the bizarre example filtering):\nclass Publication(models.Model):\n\tpass\nclass Article(models.Model):\n\tpublications = models.ManyToManyField(to=Publication, blank=True, null=True)\nclass ArticleForm(forms.ModelForm):\n\tpublications = forms.ModelMultipleChoiceField(\n\t\tPublication.objects.filter(id__lt=2) | Publication.objects.filter(id__gt=5),\n\t\trequired=False,\n\t)\n\tclass Meta:\n\t\tmodel = Article\n\t\tfields = [\"publications\"]\nclass ArticleAdmin(admin.ModelAdmin):\n\tform = ArticleForm\nThis works well. However, changing the ModelMultipleChoiceField queryset to use union() breaks things.\npublications = forms.ModelMultipleChoiceField(\n\tPublication.objects.filter(id__lt=2).union(\n\t\tPublication.objects.filter(id__gt=5)\n\t),\n\trequired=False,\n)\nThe form correctly shows only the matching objects. However, if you submit this form while empty (i.e. you didn't select any publications), ALL objects matching the queryset will be added. Using the OR query, NO objects are added, as I'd expect.\n",
  "answer": "{\"task_id\": \"39c1f27e-0b70-4533-9bba-2336dd5f295e\", \"workflow_run_id\": \"48520022-1ac7-40a2-867f-f4eabe5b5168\", \"data\": {\"id\": \"48520022-1ac7-40a2-867f-f4eabe5b5168\", \"workflow_id\": \"bc5a1a30-852f-4fd7-98f0-8fca2f23c564\", \"status\": \"succeeded\", \"outputs\": {\"ranked_patches\": \"```json\\n{\\n  \\\"ranked_patches\\\": [\\n    {\\n      \\\"patch_id\\\": 2,\\n      \\\"patch\\\": \\\"diff --git a/django/db/models/query.py b/django/db/models/query.py\\\\nindex a1b2c3d..j7k8l9m 100644\\\\n--- a/django/db/models/query.py\\\\n+++ b/django/db/models/query.py\\\\n@@ -904,6 +904,11 @@ class QuerySet:\\\\n         \\\\\\\"\\\\\\\"\\\\\\\"\\\\n         Creates an empty QuerySet.\\\\n         \\\\\\\"\\\\\\\"\\\\\\\"\\\\n+        if hasattr(self.query, \\\\\\\"combinator\\\\\\\") and self.query.combinator:\\\\n+            # In case of a combination (like union), reset to an empty result set\\\\n+            clone = self._clone()\\\\n+            clone.query.combined_queries = []\\\\n+            return clone\\\\n         clone = self._clone()\\\\n         clone._result_cache = []\\\\n         clone._sticky_filter = False\\\\n\\\\n@@ -860,6 +865,7 @@ class QuerySet:\\\\n             raise TypeError(\\\\\\\"Cannot combine queries on different base models.\\\\\\\")\\\\n\\\\n         combined = self._chain()\\\\n+        combined.query.combinator = \\\\\\\"union\\\\\\\"  # Track the query type for handling changes\\\\n         combined.query.add_q(lhs)\\\\n         combined.query.add_q(rhs)\\\\n         return combined\\\\n\\\",\\n      \\\"rank\\\": 1,\\n      \\\"score\\\": 0.9,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.9,\\n        \\\"code_quality\\\": 0.9,\\n        \\\"risk_level\\\": 0.1,\\n        \\\"best_practices\\\": 0.9,\\n        \\\"test_coverage\\\": 0.85\\n      },\\n      \\\"reasoning\\\": \\\"This patch correctly identifies and handles the case where a combined query is used, ensuring that none() returns an empty result set. It uses a clear and maintainable approach by checking for the combinator attribute and resetting the combined_queries list.\\\",\\n      \\\"recommendation\\\": \\\"Highly recommended\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 1,\\n      \\\"patch\\\": \\\"diff --git a/django/db/models/query.py b/django/db/models/query.py\\\\nindex a1b2c3d..e4f5g6h 100644\\\\n--- a/django/db/models/query.py\\\\n+++ b/django/db/models/query.py\\\\n@@ -904,6 +904,10 @@ class QuerySet:\\\\n         \\\\\\\"\\\\\\\"\\\\\\\"\\\\n         Creates an empty QuerySet.\\\\n         \\\\\\\"\\\\\\\"\\\\\\\"\\\\n+        if self.query.combinator:  # Check if the queryset is a result of union, intersection or difference\\\\n+            return self._clone()._combinator_query(None)\\\\n+\\\\n         clone = self._clone()\\\\n         clone._result_cache = []\\\\n         clone._sticky_filter = False\\\\n\\\\n@@ -860,6 +864,9 @@ class QuerySet:\\\\n             raise TypeError(\\\\\\\"Cannot combine queries on different base models.\\\\\\\")\\\\n\\\\n         combined = self._chain()\\\\n+        # Set the combinator type, this will help in none to reset the state\\\\n+        combined.query.combinator = \\\\\\\"union\\\\\\\"\\\\n+\\\\n         combined.query.add_q(lhs)\\\\n         combined.query.add_q(rhs)\\\\n         return combined\\\\n\\\",\\n      \\\"rank\\\": 2,\\n      \\\"score\\\": 0.85,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.85,\\n        \\\"code_quality\\\": 0.85,\\n        \\\"risk_level\\\": 0.15,\\n        \\\"best_practices\\\": 0.85,\\n        \\\"test_coverage\\\": 0.8\\n      },\\n      \\\"reasoning\\\": \\\"This patch also addresses the issue by checking the combinator and using a method to reset the query state. However, it introduces a new method call which might not be necessary, adding slight complexity.\\\",\\n      \\\"recommendation\\\": \\\"Recommended\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 3,\\n      \\\"patch\\\": \\\"diff --git a/django/db/models/query.py b/django/db/models/query.py\\\\nindex a1b2c3d..n0p1q2r 100644\\\\n--- a/django/db/models/query.py\\\\n+++ b/django/db/models/query.py\\\\n@@ -905,6 +905,11 @@ class QuerySet:\\\\n         \\\\\\\"\\\\\\\"\\\\\\\"\\\\n         Creates an empty QuerySet.\\\\n         \\\\\\\"\\\\\\\"\\\\\\\"\\\\n+        if getattr(self.query, 'combinator', None):\\\\n+            # When none() is called on a combined query, reset as completely empty\\\\n+            clone = self._clone()\\\\n+            clone.query = self.query.chain(null=True)\\\\n+            return clone\\\\n         clone = self._clone()\\\\n         clone._result_cache = []\\\\n         clone._sticky_filter = False\\\\n@@ -860,6 +865,8 @@ class QuerySet:\\\\n             raise TypeError(\\\\\\\"Cannot combine queries on different base models.\\\\\\\")\\\\n\\\\n         combined = self._chain()\\\\n+        combined.query.set_combinator(\\\\\\\"union\\\\\\\")  # Ensure combinator type is set\\\\n+        return combined\\\\n\\\\n     def _chain(self, **kwargs):\\\\n         \\\\\\\"\\\\\\\"\\\\\\\"\\\\n\\\",\\n      \\\"rank\\\": 3,\\n      \\\"score\\\": 0.8,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.8,\\n        \\\"code_quality\\\": 0.8,\\n        \\\"risk_level\\\": 0.2,\\n        \\\"best_practices\\\": 0.8,\\n        \\\"test_coverage\\\": 0.75\\n      },\\n      \\\"reasoning\\\": \\\"This patch uses a different approach by chaining with a null parameter, which might be less intuitive and could introduce unexpected behavior. It is less straightforward than the other patches.\\\",\\n      \\\"recommendation\\\": \\\"Consider with caution\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 4,\\n      \\\"patch\\\": \\\"diff --git a/django/forms/models.py b/django/forms/models.py\\\\nindex x9y8z7a..b5c6d7e 100644\\\\n--- a/django/forms/models.py\\\\n+++ b/django/forms/models.py\\\\n@@ -302,6 +302,9 @@ class ModelMultipleChoiceField(ModelChoiceField):\\\\n             return []\\\\n\\\\n         if value:\\\\n+            if hasattr(self.queryset, 'query') and self.queryset.query.combinator:\\\\n+                # Don't call none() on a combined queryset, it behaves differently\\\\n+                return self.queryset.none().values_list(self.to_field_name, flat=True)\\\\n             qs = self._check_values(value)\\\\n             return list(qs.values_list(self.to_field_name, flat=True))\\\\n\\\\n\\\",\\n      \\\"rank\\\": 4,\\n      \\\"score\\\": 0.7,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.7,\\n        \\\"code_quality\\\": 0.7,\\n        \\\"risk_level\\\": 0.3,\\n        \\\"best_practices\\\": 0.7,\\n        \\\"test_coverage\\\": 0.7\\n      },\\n      \\\"reasoning\\\": \\\"This patch modifies the form field logic directly, which is not ideal as it addresses the symptom rather than the root cause in the QuerySet logic. It could lead to inconsistencies elsewhere.\\\",\\n      \\\"recommendation\\\": \\\"Not recommended\\\"\\n    }\\n  ],\\n  \\\"evaluation_summary\\\": \\\"Patch 2 is the best choice as it effectively handles the none() method for combined queries with minimal complexity and risk. Patch 1 is a close second, offering a similar solution with slightly more complexity. Patch 3 introduces a less intuitive approach, and Patch 4 addresses the issue at the form level, which is not ideal.\\\"\\n}\\n```\", \"generated_tests\": \"{\\n  \\\"reproduction_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_reproduce_original_issue_union\\\",\\n      \\\"test_code\\\": \\\"def test_reproduce_original_issue_union():\\\\n    # Prepare test data\\\\n    publication1 = Publication.objects.create(id=1)\\\\n    publication2 = Publication.objects.create(id=2)\\\\n    publication3 = Publication.objects.create(id=6)\\\\n    article = Article.objects.create()\\\\n\\\\n    # Perform the union query with ModelMultipleChoiceField\\\\n    form = ArticleForm(data={'publications': [publication2.id, publication3.id]})\\\\n    form.full_clean()\\\\n    form.save(commit=False)\\\\n\\\\n    # Validate that only selected publications are associated with the article\\\\n    assert article.publications.count() == 1\\\\n    assert publication2 in article.publications.all()\\\\n\\\",\\n      \\\"description\\\": \\\"This test reproduces the original issue by submitting the form with an empty selection after applying the union() query\\\",\\n      \\\"expected_behavior\\\": \\\"The test should fail before applying the patch, as all objects matching the queryset will be added instead of an empty selection.\\\"\\n    }\\n  ],\\n  \\\"validation_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation_union\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation_union():\\\\n    # Prepare test data\\\\n    publication1 = Publication.objects.create(id=1)\\\\n    publication2 = Publication.objects.create(id=2)\\\\n    publication3 = Publication.objects.create(id=6)\\\\n\\\\n    # Apply the patch to fix the none() behavior on union() query\\\\n    # Perform the union query with ModelMultipleChoiceField\\\\n    form = ArticleForm(data={'publications': [publication2.id, publication3.id]})\\\\n    form.full_clean()\\\\n    form.save(commit=False)\\\\n\\\\n    # Validate that only selected publications are associated with the article after patching\\\\n    assert article.publications.count() == 1\\\\n    assert publication2 in article.publications.all()\\\\n\\\",\\n      \\\"description\\\": \\\"This test validates the patch works correctly by submitting the form with an empty selection after applying the union() query\\\",\\n      \\\"expected_behavior\\\": \\\"The test should pass after applying the patch, as only the selected publications should be associated with the article.\\\"\\n    }\\n  ],\\n  \\\"test_summary\\\": \\\"Reproduction and validation tests for the original issue of QuerySet.none() on combined queries returning all results, specifically related to union() queries.\\\"\\n}\"}, \"error\": \"\", \"elapsed_time\": 503.789609, \"total_tokens\": 23488, \"total_steps\": 9, \"created_at\": 1753296391, \"finished_at\": 1753296895}}"
}