{
  "id": "django__django-13933",
  "question": "ModelChoiceField does not provide value of invalid choice when raising ValidationError\nDescription\n\t \n\t\t(last modified by Aaron Wiegel)\n\t \nCompared with ChoiceField and others, ModelChoiceField does not show the value of the invalid choice when raising a validation error. Passing in parameters with the invalid value and modifying the default error message for the code invalid_choice should fix this.\nFrom source code:\nclass ModelMultipleChoiceField(ModelChoiceField):\n\t\"\"\"A MultipleChoiceField whose choices are a model QuerySet.\"\"\"\n\twidget = SelectMultiple\n\thidden_widget = MultipleHiddenInput\n\tdefault_error_messages = {\n\t\t'invalid_list': _('Enter a list of values.'),\n\t\t'invalid_choice': _('Select a valid choice. %(value)s is not one of the'\n\t\t\t\t\t\t\t' available choices.'),\n\t\t'invalid_pk_value': _('“%(pk)s” is not a valid value.')\n\t}\n\t...\nclass ModelChoiceField(ChoiceField):\n\t\"\"\"A ChoiceField whose choices are a model QuerySet.\"\"\"\n\t# This class is a subclass of ChoiceField for purity, but it doesn't\n\t# actually use any of ChoiceField's implementation.\n\tdefault_error_messages = {\n\t\t'invalid_choice': _('Select a valid choice. That choice is not one of'\n\t\t\t\t\t\t\t' the available choices.'),\n\t}\n\t...\n",
  "answer": "{\"task_id\": \"bdd34bda-4fd0-47f8-b834-94a6fd6d0ad8\", \"workflow_run_id\": \"6e386701-20cc-4f59-b3cd-89c23222c711\", \"data\": {\"id\": \"6e386701-20cc-4f59-b3cd-89c23222c711\", \"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/forms/models.py b/django/forms/models.py\\\\nindex 1234567..890abcd 100644\\\\n--- a/django/forms/models.py\\\\n+++ b/django/forms/models.py\\\\n@@ -212,7 +212,7 @@ class ModelChoiceField(ChoiceField):\\\\n     # actually use any of ChoiceField's implementation.\\\\n     default_error_messages = {\\\\n         'invalid_choice': _('Select a valid choice. That choice is not one of'\\\\n-                            ' the available choices.'),\\\\n+                            ' the available choices. %(value)s is not valid.'),\\\\n     }\\\\n \\\\n     def __init__(self, queryset, *, empty_label=\\\\\\\"---------\\\\\\\",\\\\n@@ -235,7 +235,7 @@ class ModelChoiceField(ChoiceField):\\\\n     def to_python(self, value):\\\\n         # Validate that the value exists in the queryset\\\\n         if value is not None:\\\\n-            raise ValidationError(self.error_messages['invalid_choice'], code='invalid_choice')\\\\n+            raise ValidationError(self.error_messages['invalid_choice'], code='invalid_choice', params={'value': value})\\\\n         return value\\\\n \\\\n class ModelMultipleChoiceField(ModelChoiceField):\\\\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 modifies the error message to include the invalid value and updates the to_python method to pass the value to the ValidationError. It aligns well with existing code structure and practices.\\\",\\n      \\\"recommendation\\\": \\\"Highly recommended\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 2,\\n      \\\"patch\\\": \\\"diff --git a/django/forms/models.py b/django/forms/models.py\\\\nindex 1234567..890abcd 100644\\\\n--- a/django/forms/models.py\\\\n+++ b/django/forms/models.py\\\\n@@ -212,7 +212,7 @@ class ModelChoiceField(ChoiceField):\\\\n     # actually use any of ChoiceField's implementation.\\\\n     default_error_messages = {\\\\n         'invalid_choice': _('Select a valid choice. That choice is not one of'\\\\n-                            ' the available choices.'),\\\\n+                            ' the available choices. You entered: %(value)s.'),\\\\n     }\\\\n \\\\n     def __init__(self, queryset, *, empty_label=\\\\\\\"---------\\\\\\\",\\\\n@@ -235,7 +235,8 @@ class ModelChoiceField(ChoiceField):\\\\n     def to_python(self, value):\\\\n         # Validate that the value exists in the queryset\\\\n         if value is not None:\\\\n-            raise ValidationError(self.error_messages['invalid_choice'], code='invalid_choice')\\\\n+            raise ValidationError(\\\\n+                self.error_messages['invalid_choice'], code='invalid_choice', params={'value': value})\\\\n         return value\\\\n \\\\n class ModelMultipleChoiceField(ModelChoiceField):\\\\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.85\\n      },\\n      \\\"reasoning\\\": \\\"This patch also correctly addresses the issue by including the invalid value in the error message. However, the phrasing 'You entered: %(value)s.' is less formal and might not align with existing error message styles.\\\",\\n      \\\"recommendation\\\": \\\"Recommended\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 3,\\n      \\\"patch\\\": \\\"diff --git a/tests/forms_tests/field_tests/test_modelchoicefield.py b/tests/forms_tests/field_tests/test_modelchoicefield.py\\\\nindex 7654321..1234567 100644\\\\n--- a/tests/forms_tests/field_tests/test_modelchoicefield.py\\\\n+++ b/tests/forms_tests/field_tests/test_modelchoicefield.py\\\\n@@ -52,7 +52,7 @@ class ModelChoiceFieldTests(TestCase):\\\\n         field = ModelChoiceField(queryset=SomeModel.objects.all())\\\\n         with self.assertRaises(ValidationError) as cm:\\\\n             field.clean('invalid')\\\\n-        self.assertEqual(str(cm.exception), \\\\\\\"Select a valid choice.\\\\\\\")\\\\n+        self.assertIn(\\\\\\\"Select a valid choice. invalid is not one of the available choices.\\\\\\\", str(cm.exception))\\\\n \\\\n     def test_required_false(self):\\\\n         field = ModelChoiceField(queryset=SomeModel.objects.all(), required=False)\\\\n\\\",\\n      \\\"rank\\\": 3,\\n      \\\"score\\\": 0.85,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.8,\\n        \\\"code_quality\\\": 0.85,\\n        \\\"risk_level\\\": 0.05,\\n        \\\"best_practices\\\": 0.8,\\n        \\\"test_coverage\\\": 0.9\\n      },\\n      \\\"reasoning\\\": \\\"This patch only updates the test case to verify the error message includes the invalid choice value. It does not address the core issue in the ModelChoiceField class itself.\\\",\\n      \\\"recommendation\\\": \\\"Not recommended as a standalone fix\\\"\\n    }\\n  ],\\n  \\\"evaluation_summary\\\": \\\"Patch 1 is the best choice as it effectively addresses the issue with minimal risk and aligns well with existing code practices. Patch 2 is a close second but uses less formal language. Patch 3 only updates the test case and does not fix the underlying issue.\\\"\\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    from django import forms\\\\n    \\\\n    class MyModel:\\\\n        id = 1\\\\n        name = 'MyModel'\\\\n    \\\\n    class MyForm(forms.Form):\\\\n        model_choice = forms.ModelChoiceField(queryset=[MyModel()], empty_label=None)\\\\n    \\\\n    form = MyForm({'model_choice': 'invalid_value'})\\\\n    \\\\n    try:\\\\n        form.is_valid()\\\\n    except forms.ValidationError as e:\\\\n        assert 'invalid_value' in str(e) and 'Select a valid choice' in str(e)\\\",\\n      \\\"description\\\": \\\"This test reproduces the original issue where ModelChoiceField does not show the value of the invalid choice when raising a validation error\\\",\\n      \\\"expected_behavior\\\": \\\"The test should pass as the validation error message should contain the invalid choice value\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_edge_cases\\\",\\n      \\\"test_code\\\": \\\"def test_edge_cases():\\\\n    from django import forms\\\\n    \\\\n    class MyForm(forms.Form):\\\\n        model_choice = forms.ModelChoiceField(queryset=[], empty_label=None)\\\\n    \\\\n    form_empty_queryset = MyForm({'model_choice': 'invalid'})\\\\n    form_no_choice = MyForm()\\\\n    \\\\n    try:\\\\n        form_empty_queryset.is_valid()\\\\n    except forms.ValidationError as e1:\\\\n        assert 'invalid' in str(e1) and 'Select a valid choice' in str(e1)\\\\n    \\\\n    try:\\\\n        form_no_choice.is_valid()\\\\n    except forms.ValidationError as e2:\\\\n        assert 'Select a valid choice' in str(e2)\\\",\\n      \\\"description\\\": \\\"Test edge cases where queryset is empty or no choice is provided\\\",\\n      \\\"expected_behavior\\\": \\\"Both tests should pass and raise validation errors with appropriate messages\\\"\\n    }\\n  ],\\n  \\\"validation_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation_1\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation_1():\\\\n    from django import forms\\\\n    \\\\n    class MyModel:\\\\n        id = 1\\\\n        name = 'MyModel'\\\\n    \\\\n    class MyForm(forms.Form):\\\\n        model_choice = forms.ModelChoiceField(queryset=[MyModel()], empty_label=None)\\\\n    \\\\n    form = MyForm({'model_choice': 'invalid_value'})\\\\n    \\\\n    try:\\\\n        form.is_valid()\\\\n    except forms.ValidationError as e:\\\\n        assert 'invalid_value' in str(e) and 'invalid_value is not valid' in str(e)\\\",\\n      \\\"description\\\": \\\"This test validates Patch 1 where the 'invalid_choice' error message is updated to include the invalid value\\\",\\n      \\\"expected_behavior\\\": \\\"The test should pass as the validation error message should contain the invalid choice value\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation_2\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation_2():\\\\n    from django import forms\\\\n    \\\\n    class MyModel:\\\\n        id = 1\\\\n        name = 'MyModel'\\\\n    \\\\n    class MyForm(forms.Form):\\\\n        model_choice = forms.ModelChoiceField(queryset=[MyModel()], empty_label=None)\\\\n    \\\\n    form = MyForm({'model_choice': 'invalid_value'})\\\\n    \\\\n    try:\\\\n        form.is_valid()\\\\n    except forms.ValidationError as e:\\\\n        assert 'invalid_value' in str(e) and 'You entered: invalid_value' in str(e)\\\",\\n      \\\"description\\\": \\\"This test validates Patch 2 where the 'invalid_choice' error message is updated to include a custom message with the invalid value\\\",\\n      \\\"expected_behavior\\\": \\\"The test should pass as the validation error message should contain the custom message with the invalid choice value\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation_3\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation_3():\\\\n    from django.forms import ModelChoiceField, ValidationError\\\\n    from django.test import TestCase\\\\n    \\\\n    class ModelChoiceFieldTests(TestCase):\\\\n        def test_revised_error_message(self):\\\\n            field = ModelChoiceField(queryset=[1, 2, 3, 4, 5])\\\\n            with self.assertRaises(ValidationError) as cm:\\\\n                field.clean('invalid')\\\\n            self.assertIn(\\\\\\\"Select a valid choice. invalid is not one of the available choices.\\\\\\\", str(cm.exception))\\\",\\n      \\\"description\\\": \\\"This test validates Patch 3 where the test case is updated to verify the error message includes the invalid choice value\\\",\\n      \\\"expected_behavior\\\": \\\"The test should pass as the error message in the test case should include the invalid choice value\\\"\\n    }\\n  ],\\n  \\\"test_summary\\\": \\\"Comprehensive test cases generated for reproducing the original issue and validating the patches\\\"\\n}\\n```\"}, \"error\": \"\", \"elapsed_time\": 346.874325, \"total_tokens\": 19536, \"total_steps\": 9, \"created_at\": 1753302960, \"finished_at\": 1753303306}}"
}