{
  "id": "django__django-11964",
  "question": "The value of a TextChoices/IntegerChoices field has a differing type\nDescription\n\t\nIf we create an instance of a model having a CharField or IntegerField with the keyword choices pointing to IntegerChoices or TextChoices, the value returned by the getter of the field will be of the same type as the one created by enum.Enum (enum value).\nFor example, this model:\nfrom django.db import models\nfrom django.utils.translation import gettext_lazy as _\nclass MyChoice(models.TextChoices):\n\tFIRST_CHOICE = \"first\", _(\"The first choice, it is\")\n\tSECOND_CHOICE = \"second\", _(\"The second choice, it is\")\nclass MyObject(models.Model):\n\tmy_str_value = models.CharField(max_length=10, choices=MyChoice.choices)\nThen this test:\nfrom django.test import TestCase\nfrom testing.pkg.models import MyObject, MyChoice\nclass EnumTest(TestCase):\n\tdef setUp(self) -> None:\n\t\tself.my_object = MyObject.objects.create(my_str_value=MyChoice.FIRST_CHOICE)\n\tdef test_created_object_is_str(self):\n\t\tmy_object = self.my_object\n\t\tself.assertIsInstance(my_object.my_str_value, str)\n\t\tself.assertEqual(str(my_object.my_str_value), \"first\")\n\tdef test_retrieved_object_is_str(self):\n\t\tmy_object = MyObject.objects.last()\n\t\tself.assertIsInstance(my_object.my_str_value, str)\n\t\tself.assertEqual(str(my_object.my_str_value), \"first\")\nAnd then the results:\n(django30-venv) ➜ django30 ./manage.py test\nCreating test database for alias 'default'...\nSystem check identified no issues (0 silenced).\nF.\n======================================================================\nFAIL: test_created_object_is_str (testing.tests.EnumTest)\n----------------------------------------------------------------------\nTraceback (most recent call last):\n File \"/Users/mikailkocak/Development/django30/testing/tests.py\", line 14, in test_created_object_is_str\n\tself.assertEqual(str(my_object.my_str_value), \"first\")\nAssertionError: 'MyChoice.FIRST_CHOICE' != 'first'\n- MyChoice.FIRST_CHOICE\n+ first\n----------------------------------------------------------------------\nRan 2 tests in 0.002s\nFAILED (failures=1)\nWe notice when invoking __str__(...) we don't actually get the value property of the enum value which can lead to some unexpected issues, especially when communicating to an external API with a freshly created instance that will send MyEnum.MyValue, and the one that was retrieved would send my_value.\n",
  "answer": "{\"task_id\": \"d7962ee0-a2cb-4ed3-9bfd-7e44cb464fd9\", \"workflow_run_id\": \"eaffa88b-f2ea-45fb-8b2b-95a49663602c\", \"data\": {\"id\": \"eaffa88b-f2ea-45fb-8b2b-95a49663602c\", \"workflow_id\": \"bc5a1a30-852f-4fd7-98f0-8fca2f23c564\", \"status\": \"succeeded\", \"outputs\": {\"ranked_patches\": \"```json\\n{\\n  \\\"ranked_patches\\\": [\\n    {\\n      \\\"patch_id\\\": 3,\\n      \\\"patch\\\": \\\"diff --git a/django/db/models/fields/__init__.py b/django/db/models/fields/__init__.py\\\\nindex 1234567..89abcde 100644\\\\n--- a/django/db/models/fields/__init__.py\\\\n+++ b/django/db/models/fields/__init__.py\\\\n@@ -605,6 +605,12 @@ class Field:\\\\n     def __set__(self, instance, value):\\\\n         if isinstance(value, str):\\\\n             instance.__dict__[self.name] = value\\\\n+        elif isinstance(value, enum.Enum):\\\\n+            # Use a helper function to convert enum member to its value\\\\n+            instance.__dict__[self.name] = self._get_enum_value(value)\\\\n         else:\\\\n             instance.__dict__[self.name] = value\\\\n+\\\\n+    def _get_enum_value(self, enum_member):\\\\n+        \\\\\\\"\\\\\\\"\\\\\\\"Helper method to retrieve the value of an enum member.\\\\\\\"\\\\\\\"\\\\\\\"\\\\n+        return enum_member.value\\\\n\\\",\\n      \\\"rank\\\": 1,\\n      \\\"score\\\": 0.95,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.95,\\n        \\\"code_quality\\\": 0.95,\\n        \\\"risk_level\\\": 0.1,\\n        \\\"best_practices\\\": 0.95,\\n        \\\"test_coverage\\\": 0.9\\n      },\\n      \\\"reasoning\\\": \\\"This patch uses a helper function to encapsulate the conversion logic, which improves code readability and maintainability. It correctly addresses the issue with minimal risk of introducing new bugs.\\\",\\n      \\\"recommendation\\\": \\\"Highly recommended\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 1,\\n      \\\"patch\\\": \\\"diff --git a/django/db/models/fields/__init__.py b/django/db/models/fields/__init__.py\\\\nindex 1234567..89abcde 100644\\\\n--- a/django/db/models/fields/__init__.py\\\\n+++ b/django/db/models/fields/__init__.py\\\\n@@ -605,6 +605,10 @@ class Field:\\\\n     def __set__(self, instance, value):\\\\n         if isinstance(value, str):\\\\n             instance.__dict__[self.name] = value\\\\n+        elif isinstance(value, enum.Enum):\\\\n+            # Convert enum member to its underlying value\\\\n+            instance.__dict__[self.name] = value.value\\\\n+         else:\\\\n+             instance.__dict__[self.name] = value\\\\n\\\",\\n      \\\"rank\\\": 2,\\n      \\\"score\\\": 0.9,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.9,\\n        \\\"code_quality\\\": 0.85,\\n        \\\"risk_level\\\": 0.15,\\n        \\\"best_practices\\\": 0.85,\\n        \\\"test_coverage\\\": 0.85\\n      },\\n      \\\"reasoning\\\": \\\"This patch directly converts enum members to their primitive values in the __set__ method. It is a straightforward solution but lacks the encapsulation that improves maintainability.\\\",\\n      \\\"recommendation\\\": \\\"Recommended\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 2,\\n      \\\"patch\\\": \\\"diff --git a/django/db/models/fields/__init__.py b/django/db/models/fields/__init__.py\\\\nindex 1234567..89abcde 100644\\\\n--- a/django/db/models/fields/__init__.py\\\\n+++ b/django/db/models/fields/__init__.py\\\\n@@ -603,7 +603,14 @@ class Field:\\\\n \\\\n     def __set__(self, instance, value):\\\\n         if isinstance(value, str):\\\\n-            instance.__dict__[self.name] = value\\\\n+            instance.__dict__[self.name] = value\\\\n+        elif isinstance(value, enum.Enum):\\\\n+            # Convert enum member to its underlying value\\\\n+            instance.__dict__[self.name] = value.value\\\\n         else:\\\\n             instance.__dict__[self.name] = value\\\\n+        # Log debug information to trace value type and assignment\\\\n+        logger.debug(f\\\\\\\"Setting {self.name} to {instance.__dict__[self.name]} (type: {type(instance.__dict__[self.name])})\\\\\\\")\\\\n\\\",\\n      \\\"rank\\\": 3,\\n      \\\"score\\\": 0.85,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.85,\\n        \\\"code_quality\\\": 0.8,\\n        \\\"risk_level\\\": 0.2,\\n        \\\"best_practices\\\": 0.8,\\n        \\\"test_coverage\\\": 0.8\\n      },\\n      \\\"reasoning\\\": \\\"This patch adds logging to track value assignments for debugging purposes. While useful for debugging, it introduces additional complexity and potential performance overhead without directly improving the core functionality.\\\",\\n      \\\"recommendation\\\": \\\"Consider if debugging is necessary\\\"\\n    }\\n  ],\\n  \\\"evaluation_summary\\\": \\\"Patch 3 is the best choice due to its use of a helper function, which enhances code maintainability and readability. Patch 1 is a straightforward solution but lacks encapsulation. Patch 2 adds logging, which is useful for debugging but adds complexity and potential overhead.\\\"\\n}\\n```\", \"generated_tests\": \"{\\n  \\\"reproduction_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_reproduce_original_issue\\\",\\n      \\\"test_code\\\": \\\"def test_reproduce_original_issue():\\\\n    choice_value = MyChoice.FIRST_CHOICE\\\\n    assert str(choice_value) == 'first', 'Enum value does not match expected string value'\\\\n    my_object = MyObject.objects.create(my_str_value=choice_value)\\\\n    assert str(my_object.my_str_value) == 'MyChoice.FIRST_CHOICE', 'Incorrect value set in the model'\\\\n    retrieved_object = MyObject.objects.last()\\\\n    assert str(retrieved_object.my_str_value) == 'MyChoice.FIRST_CHOICE', 'Incorrect value retrieved from the model'\\\",\\n      \\\"description\\\": \\\"Reproduces the original issue where enum value is not correctly converted to string value in Django model field\\\",\\n      \\\"expected_behavior\\\": \\\"The test should fail before the patch and pass after applying the patch\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_edge_cases\\\",\\n      \\\"test_code\\\": \\\"def test_edge_cases():\\\\n    choice_value = MyChoice.FIRST_CHOICE\\\\n    assert isinstance(choice_value, MyChoice), 'Choice value is not an instance of MyChoice'\\\\n    my_object_empty = MyObject()\\\\n    assert my_object_empty.my_str_value is None, 'Initial value of my_str_value is not None'\\\\n    my_object_invalid_choice = MyObject(my_str_value='invalid_choice')\\\\n    assert my_object_invalid_choice.my_str_value == 'invalid_choice', 'Invalid choice value is not stored in my_str_value'\\\",\\n      \\\"description\\\": \\\"Tests edge cases related to the issue such as empty initial value and setting invalid choice value\\\",\\n      \\\"expected_behavior\\\": \\\"The test cases should cover edge cases and handle accordingly\\\"\\n    }\\n  ],\\n  \\\"validation_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation_1\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation_1():\\\\n    choice_value = MyChoice.FIRST_CHOICE\\\\n    assert str(choice_value) == 'first', 'Enum value does not match expected string value'\\\\n    my_object = MyObject.objects.create(my_str_value=choice_value)\\\\n    assert str(my_object.my_str_value) == 'first', 'Correct value not set in the model after applying patch 1'\\\",\\n      \\\"description\\\": \\\"Validates the first patch that directly converts enum members to their primitive values in the __set__ method\\\",\\n      \\\"expected_behavior\\\": \\\"The test should pass confirming the correct conversion of enum values\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation_2\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation_2():\\\\n    choice_value = MyChoice.FIRST_CHOICE\\\\n    assert str(choice_value) == 'first', 'Enum value does not match expected string value'\\\\n    my_object = MyObject.objects.create(my_str_value=choice_value)\\\\n    assert str(my_object.my_str_value) == 'first', 'Correct value not set in the model after applying patch 2'\\\",\\n      \\\"description\\\": \\\"Validates the second patch that adds logging to track value assignments for debugging purposes\\\",\\n      \\\"expected_behavior\\\": \\\"The test should pass confirming the correct logging implementation\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation_3\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation_3():\\\\n    choice_value = MyChoice.FIRST_CHOICE\\\\n    assert str(choice_value) == 'first', 'Enum value does not match expected string value'\\\\n    my_object = MyObject.objects.create(my_str_value=choice_value)\\\\n    assert str(my_object.my_str_value) == 'first', 'Correct value not set in the model after applying patch 3'\\\",\\n      \\\"description\\\": \\\"Validates the third patch that uses a helper function to convert enum member to its value\\\",\\n      \\\"expected_behavior\\\": \\\"The test should pass confirming the correct usage of the helper function for conversion\\\"\\n    }\\n  ],\\n  \\\"test_summary\\\": \\\"Comprehensive test cases have been generated to reproduce the original issue, cover edge cases, and validate the patches for fixing the issue.\\\"\\n}\"}, \"error\": \"\", \"elapsed_time\": 401.803295, \"total_tokens\": 23866, \"total_steps\": 9, \"created_at\": 1753288488, \"finished_at\": 1753288890}}"
}