{
  "id": "django__django-11211",
  "question": "Prefetch related is not working when used GFK for model that uses UUID field as PK.\nDescription\n\t\nHow to reproduce:\ncreate model with UUID as primary key\nclass Foo(models.Model):\n\tid = models.UUIDField(primary_key=True, default=uuid.uuid4, editable=False)\n\t...\ncreate another model with GFK to model Foo\nclass Bar(models.Model):\n\tfoo_content_type = models.ForeignKey(\n\t\tContentType, related_name='actor',\n\t\ton_delete=models.CASCADE, db_index=True\n\t)\n\tfoo_object_id = models.CharField(max_length=255, db_index=True)\n\tfoo = GenericForeignKey('foo_content_type', 'foo_object_id')\n\t...\nand try to get queryset with prefetch related (django orm engine return None for attribute foo):\nBar.objects.all().prefetch_related('foo')\nThanks a lot for your attention! Also i wanna point out some related bug report from third party library in which previously i faced with that issue, maybe it would useful – ​https://github.com/justquick/django-activity-stream/issues/245\n",
  "answer": "{\"task_id\": \"1407ece6-bc62-4e73-b71e-815b99978fb1\", \"workflow_run_id\": \"220b70c7-5bdc-4eea-9dd2-8073dddedaab\", \"data\": {\"id\": \"220b70c7-5bdc-4eea-9dd2-8073dddedaab\", \"workflow_id\": \"9a0a6efa-4d15-4876-b13f-4fb6a999caa2\", \"status\": \"succeeded\", \"outputs\": {\"final_patch\": [\"--- a/django/contrib/contenttypes/fields.py\\n+++ b/django/contrib/contenttypes/fields.py\\n@@ -470,6 +470,16 @@ class GenericForeignKey(FieldCacheMixin):\\n         # Group the instances by content type.\\n         content_type_queries = {}\\n         for ct_id, fkey in ct_attname_to_model.items():\\n+            # Get the target model's primary key field to cast object_id values\\n+            target_model = ct_id_to_model[ct_id]\\n+            pk_field = target_model._meta.pk\\n+            \\n+            # Cast object_id values to the correct type\\n+            cast_fkeys = []\\n+            for fk in fkey:\\n+                cast_fkeys.append(pk_field.to_python(fk))\\n+            fkey = cast_fkeys\\n+            \\n             content_type_queries[ct_id] = (\\n                 ct_id_to_model[ct_id]._base_manager.filter(\\n                     pk__in=fkey\\n@@ -477,6 +487,7 @@ class GenericForeignKey(FieldCacheMixin):\\n                 fkey\\n             )\\n \\n+        # Rest of the method remains the same\\n         ret_val = []\\n         for ct_id, objs in content_type_queries.items():\\n             ret_val.extend(objs)\", \"--- a/django/contrib/contenttypes/fields.py\\n+++ b/django/contrib/contenttypes/fields.py\\n@@ -460,6 +460,20 @@ class GenericForeignKey(FieldCacheMixin):\\n         # Group instances by content type.\\n         ct_attname_to_model = defaultdict(list)\\n         for instance in instances:\\n+            # Get the content type and target model\\n+            ct_id = getattr(instance, ct_attname)\\n+            if ct_id is not None:\\n+                try:\\n+                    target_model = ct_id_to_model[ct_id]\\n+                    pk_field = target_model._meta.pk\\n+                    # Cast the object_id to the correct type\\n+                    obj_id = getattr(instance, self.fk_field)\\n+                    if obj_id is not None:\\n+                        obj_id = pk_field.to_python(obj_id)\\n+                except (KeyError, ValueError, TypeError):\\n+                    # Skip instances with invalid content types or object IDs\\n+                    continue\\n+                ct_attname_to_model[ct_id].append(obj_id)\\n-            ct_attname_to_model[\\n-                getattr(instance, ct_attname)\\n-            ].append(getattr(instance, self.fk_field))\\n \\n         # Group the instances by content type.\\n         content_type_queries = {}\", \"--- a/django/contrib/contenttypes/fields.py\\n+++ b/django/contrib/contenttypes/fields.py\\n@@ -470,8 +470,18 @@ class GenericForeignKey(FieldCacheMixin):\\n         # Group the instances by content type.\\n         content_type_queries = {}\\n         for ct_id, fkey in ct_attname_to_model.items():\\n+            # Get the target model and its primary key field\\n+            target_model = ct_id_to_model[ct_id]\\n+            pk_field = target_model._meta.pk\\n+            \\n+            # Convert object_id values to the correct type, filtering out None values\\n+            converted_fkeys = []\\n+            for obj_id in fkey:\\n+                if obj_id is not None:\\n+                    converted_fkeys.append(pk_field.to_python(obj_id))\\n+            \\n             content_type_queries[ct_id] = (\\n-                ct_id_to_model[ct_id]._base_manager.filter(\\n-                    pk__in=fkey\\n+                target_model._base_manager.filter(\\n+                    pk__in=converted_fkeys\\n                 ),\\n-                fkey\\n+                converted_fkeys\\n             )\", \"--- a/django/contrib/contenttypes/fields.py\\n+++ b/django/contrib/contenttypes/fields.py\\n@@ -440,6 +440,18 @@ class GenericForeignKey(FieldCacheMixin):\\n         return self.remote_field.model._meta.get_field(self.object_id_field_name)\\n \\n     def get_prefetch_queryset(self, instances, queryset=None):\\n+        def _convert_object_ids(object_ids, target_model):\\n+            \\\"\\\"\\\"Convert object_id values to the correct type for the target model's PK.\\\"\\\"\\\"\\n+            pk_field = target_model._meta.pk\\n+            converted_ids = []\\n+            for obj_id in object_ids:\\n+                if obj_id is not None:\\n+                    try:\\n+                        converted_ids.append(pk_field.to_python(obj_id))\\n+                    except (ValueError, TypeError):\\n+                        # Skip invalid object IDs\\n+                        continue\\n+            return converted_ids\\n+        \\n         if queryset is not None:\\n             raise ValueError(\\\"Custom queryset can't be used for this lookup.\\\")\\n \\n@@ -470,6 +482,9 @@ class GenericForeignKey(FieldCacheMixin):\\n         # Group the instances by content type.\\n         content_type_queries = {}\\n         for ct_id, fkey in ct_attname_to_model.items():\\n+            target_model = ct_id_to_model[ct_id]\\n+            # Convert object_id values to the correct type\\n+            fkey = _convert_object_ids(fkey, target_model)\\n             content_type_queries[ct_id] = (\\n-                ct_id_to_model[ct_id]._base_manager.filter(\\n+                target_model._base_manager.filter(\\n                     pk__in=fkey\\n                 ),\", \"--- a/django/contrib/contenttypes/fields.py\\n+++ b/django/contrib/contenttypes/fields.py\\n@@ -472,6 +472,11 @@ class GenericForeignKey(FieldCacheMixin):\\n         for ct_id, fkey in ct_attname_to_model.items():\\n+            # Convert object_id values to match the target model's primary key type\\n+            target_model = ct_id_to_model[ct_id]\\n+            pk_field = target_model._meta.pk\\n+            fkey = [pk_field.to_python(obj_id) for obj_id in fkey if obj_id is not None]\\n+            \\n             content_type_queries[ct_id] = (\\n-                ct_id_to_model[ct_id]._base_manager.filter(\\n+                target_model._base_manager.filter(\\n                     pk__in=fkey\\n                 ),\\n                 fkey\", \"diff\\n--- a/django/contrib/contenttypes/fields.py\\n+++ b/django/contrib/contenttypes/fields.py\\n@@ -460,9 +460,15 @@ class GenericForeignKey(FieldCacheMixin):\\n         # Group instances by content type.\\n         ct_attname_to_model = defaultdict(list)\\n         for instance in instances:\\n-            ct_attname_to_model[\\n-                getattr(instance, ct_attname)\\n-            ].append(getattr(instance, self.fk_field))\\n+            ct_id = getattr(instance, ct_attname)\\n+            obj_id = getattr(instance, self.fk_field)\\n+            if ct_id is not None and obj_id is not None:\\n+                # Convert object_id to the correct type for the target model's PK\\n+                try:\\n+                    target_model = ct_id_to_model[ct_id]\\n+                    pk_field = target_model._meta.pk\\n+                    obj_id = pk_field.to_python(obj_id)\\n+                    ct_attname_to_model[ct_id].append(obj_id)\\n+                except (KeyError, ValueError, TypeError):\\n+                    # Skip instances with invalid content types or object IDs\\n+                    continue\\n \\n         # Group the instances by content type.\\n         content_type_queries = {}\\n\", \"diff\\n--- a/django/contrib/contenttypes/fields.py\\n+++ b/django/contrib/contenttypes/fields.py\\n@@ -460,9 +460,15 @@ class GenericForeignKey(FieldCacheMixin):\\n         # Group instances by content type.\\n         ct_attname_to_model = defaultdict(list)\\n         for instance in instances:\\n-            ct_attname_to_model[\\n-                getattr(instance, ct_attname)\\n-            ].append(getattr(instance, self.fk_field))\\n+            ct_id = getattr(instance, ct_attname)\\n+            obj_id = getattr(instance, self.fk_field)\\n+            if ct_id is not None and obj_id is not None:\\n+                # Convert object_id to the correct type for the target model's PK\\n+                try:\\n+                    target_model = ct_id_to_model[ct_id]\\n+                    pk_field = target_model._meta.pk\\n+                    obj_id = pk_field.to_python(obj_id)\\n+                    ct_attname_to_model[ct_id].append(obj_id)\\n+                except (KeyError, ValueError, TypeError):\\n+                    # Skip instances with invalid content types or object IDs\\n+                    continue\\n \\n         # Group the instances by content type.\\n         content_type_queries = {}\\n\", \"diff\\n--- a/django/contrib/contenttypes/fields.py\\n+++ b/django/contrib/contenttypes/fields.py\\n@@ -440,6 +440,18 @@ class GenericForeignKey(FieldCacheMixin):\\n         return self.remote_field.model._meta.get_field(self.object_id_field_name)\\n \\n     def get_prefetch_queryset(self, instances, queryset=None):\\n+        def _convert_object_ids(object_ids, target_model):\\n+            \\\"\\\"\\\"Convert object_id values to the correct type for the target model's PK.\\\"\\\"\\\"\\n+            pk_field = target_model._meta.pk\\n+            converted_ids = []\\n+            for obj_id in object_ids:\\n+                if obj_id is not None:\\n+                    try:\\n+                        converted_ids.append(pk_field.to_python(obj_id))\\n+                    except (ValueError, TypeError):\\n+                        # Skip invalid object IDs\\n+                        continue\\n+            return converted_ids\\n+        \\n         if queryset is not None:\\n             raise ValueError(\\\"Custom queryset can't be used for this lookup.\\\")\\n \\n@@ -470,6 +482,9 @@ class GenericForeignKey(FieldCacheMixin):\\n         # Group the instances by content type.\\n         content_type_queries = {}\\n         for ct_id, fkey in ct_attname_to_model.items():\\n+            target_model = ct_id_to_model[ct_id]\\n+            # Convert object_id values to the correct type\\n+            fkey = _convert_object_ids(fkey, target_model)\\n             content_type_queries[ct_id] = (\\n-                ct_id_to_model[ct_id]._base_manager.filter(\\n+                target_model._base_manager.filter(\\n                     pk__in=fkey\\n                 ),\\n\"]}, \"error\": \"\", \"elapsed_time\": 112.251586, \"total_tokens\": 42412, \"total_steps\": 30, \"created_at\": 1754648386, \"finished_at\": 1754648498}}"
}