{
  "Selected_candidate": {
    "pr_number": 5437,
    "pr_title": "Fixed #25560 -- Made empty string related name invalid.",
    "pr_body": "Thanks to Ali Lotfi for the initial report and patch.\n",
    "issue_id": 25560,
    "issue_title": "Empty string related name should be invalid",
    "issue_body": "The actual model field check is\n​\nskipped\nif the\nrelated_name\nis an empty string when a warning should be raised since it's an invalid Python identifier.",
    "issue_closed_at": "2015-10-16T13:18:21",
    "base_commit": "4dcc2a195595f8d7ddad45bc4baf98ffdeec7f41",
    "changes": [
      {
        "file": "django/db/models/fields/related.py",
        "type": "function",
        "name": "_check_related_name_is_valid",
        "class_name": "RelatedField",
        "code": "def _check_related_name_is_valid(self):\n        import re\n        import keyword\n        related_name = self.remote_field.related_name\n        if not related_name:\n            return []\n        is_valid_id = True\n        if keyword.iskeyword(related_name):\n            is_valid_id = False\n        if six.PY3:\n            if not related_name.isidentifier():\n                is_valid_id = False\n        else:\n            if not re.match(r'^[a-zA-Z_][a-zA-Z0-9_]*\\Z', related_name):\n                is_valid_id = False\n        if not (is_valid_id or related_name.endswith('+')):\n            return [\n                checks.Error(\n                    \"The name '%s' is invalid related_name for field %s.%s\" %\n                    (self.remote_field.related_name, self.model._meta.object_name,\n                     self.name),\n                    hint=\"Related name must be a valid Python identifier or end with a '+'\",\n                    obj=self,\n                    id='fields.E306',\n                )\n            ]\n        return []"
      },
      {
        "file": "django/db/models/fields/reverse_related.py",
        "type": "function",
        "name": "get_db_prep_lookup",
        "class_name": "ForeignObjectRel",
        "code": "def get_db_prep_lookup(self, lookup_type, value, connection, prepared=False):\n        # Defer to the actual field definition for db prep\n        return self.field.get_db_prep_lookup(lookup_type, value, connection=connection, prepared=prepared)"
      }
    ]
  },
  "Justification": "Candidate E is particularly helpful because it deals with field validations like ModelChoiceField, highlighting issues in user input handling similar to the CURRENT bug. Both bugs involve validation errors that arise from invalid data being processed in a form context. Insights from how Candidate E resolves invalid related names can inform approaches for modifying error messages in ModelChoiceField. It also addresses essential checks that involve ensuring valid choices, aligning well with the current concern of reflecting invalid choices properly.",
  "instance_id": "django__django-13933",
  "repo": "django/django",
  "created_at": "2021-01-26T03:58:23Z",
  "problem_statement": "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",
  "patch": "diff --git a/django/forms/models.py b/django/forms/models.py\n--- a/django/forms/models.py\n+++ b/django/forms/models.py\n@@ -1284,7 +1284,11 @@ def to_python(self, value):\n                 value = getattr(value, key)\n             value = self.queryset.get(**{key: value})\n         except (ValueError, TypeError, self.queryset.model.DoesNotExist):\n-            raise ValidationError(self.error_messages['invalid_choice'], code='invalid_choice')\n+            raise ValidationError(\n+                self.error_messages['invalid_choice'],\n+                code='invalid_choice',\n+                params={'value': value},\n+            )\n         return value\n \n     def validate(self, value):\n"
}