mirror of
				https://github.com/graphql-python/graphene-django.git
				synced 2025-10-31 07:57:31 +03:00 
			
		
		
		
	* adding optional_fields to enforce fields to be optional * adding support for all * adding unit tests * Update graphene_django/rest_framework/mutation.py Co-authored-by: Kien Dang <kiend@pm.me> * linting * linting * add missing import --------- Co-authored-by: Kien Dang <kiend@pm.me>
		
			
				
	
	
		
			169 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			169 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| from functools import singledispatch
 | |
| 
 | |
| from django.core.exceptions import ImproperlyConfigured
 | |
| from rest_framework import serializers
 | |
| 
 | |
| import graphene
 | |
| 
 | |
| from ..converter import convert_choices_to_named_enum_with_descriptions
 | |
| from ..registry import get_global_registry
 | |
| from .types import DictType
 | |
| 
 | |
| 
 | |
| @singledispatch
 | |
| def get_graphene_type_from_serializer_field(field):
 | |
|     raise ImproperlyConfigured(
 | |
|         f"Don't know how to convert the serializer field {field} ({field.__class__}) "
 | |
|         "to Graphene type"
 | |
|     )
 | |
| 
 | |
| 
 | |
| def convert_serializer_field(
 | |
|     field, is_input=True, convert_choices_to_enum=True, force_optional=False
 | |
| ):
 | |
|     """
 | |
|     Converts a django rest frameworks field to a graphql field
 | |
|     and marks the field as required if we are creating an input type
 | |
|     and the field itself is required
 | |
|     """
 | |
| 
 | |
|     if isinstance(field, serializers.ChoiceField) and not convert_choices_to_enum:
 | |
|         graphql_type = graphene.String
 | |
|     else:
 | |
|         graphql_type = get_graphene_type_from_serializer_field(field)
 | |
| 
 | |
|     args = []
 | |
|     kwargs = {
 | |
|         "description": field.help_text,
 | |
|         "required": is_input and field.required and not force_optional,
 | |
|     }
 | |
| 
 | |
|     # if it is a tuple or a list it means that we are returning
 | |
|     # the graphql type and the child type
 | |
|     if isinstance(graphql_type, (list, tuple)):
 | |
|         kwargs["of_type"] = graphql_type[1]
 | |
|         graphql_type = graphql_type[0]
 | |
| 
 | |
|     if isinstance(field, serializers.ModelSerializer):
 | |
|         if is_input:
 | |
|             graphql_type = convert_serializer_to_input_type(field.__class__)
 | |
|         else:
 | |
|             global_registry = get_global_registry()
 | |
|             field_model = field.Meta.model
 | |
|             args = [global_registry.get_type_for_model(field_model)]
 | |
|     elif isinstance(field, serializers.ListSerializer):
 | |
|         field = field.child
 | |
|         if is_input:
 | |
|             kwargs["of_type"] = convert_serializer_to_input_type(field.__class__)
 | |
|         else:
 | |
|             del kwargs["of_type"]
 | |
|             global_registry = get_global_registry()
 | |
|             field_model = field.Meta.model
 | |
|             args = [global_registry.get_type_for_model(field_model)]
 | |
| 
 | |
|     return graphql_type(*args, **kwargs)
 | |
| 
 | |
| 
 | |
| def convert_serializer_to_input_type(serializer_class):
 | |
|     cached_type = convert_serializer_to_input_type.cache.get(
 | |
|         serializer_class.__name__, None
 | |
|     )
 | |
|     if cached_type:
 | |
|         return cached_type
 | |
|     serializer = serializer_class()
 | |
| 
 | |
|     items = {
 | |
|         name: convert_serializer_field(field)
 | |
|         for name, field in serializer.fields.items()
 | |
|     }
 | |
|     ret_type = type(
 | |
|         f"{serializer.__class__.__name__}Input",
 | |
|         (graphene.InputObjectType,),
 | |
|         items,
 | |
|     )
 | |
|     convert_serializer_to_input_type.cache[serializer_class.__name__] = ret_type
 | |
|     return ret_type
 | |
| 
 | |
| 
 | |
| convert_serializer_to_input_type.cache = {}
 | |
| 
 | |
| 
 | |
| @get_graphene_type_from_serializer_field.register(serializers.Field)
 | |
| def convert_serializer_field_to_string(field):
 | |
|     return graphene.String
 | |
| 
 | |
| 
 | |
| @get_graphene_type_from_serializer_field.register(serializers.ModelSerializer)
 | |
| def convert_serializer_to_field(field):
 | |
|     return graphene.Field
 | |
| 
 | |
| 
 | |
| @get_graphene_type_from_serializer_field.register(serializers.ListSerializer)
 | |
| def convert_list_serializer_to_field(field):
 | |
|     child_type = get_graphene_type_from_serializer_field(field.child)
 | |
|     return (graphene.List, child_type)
 | |
| 
 | |
| 
 | |
| @get_graphene_type_from_serializer_field.register(serializers.IntegerField)
 | |
| def convert_serializer_field_to_int(field):
 | |
|     return graphene.Int
 | |
| 
 | |
| 
 | |
| @get_graphene_type_from_serializer_field.register(serializers.BooleanField)
 | |
| def convert_serializer_field_to_bool(field):
 | |
|     return graphene.Boolean
 | |
| 
 | |
| 
 | |
| @get_graphene_type_from_serializer_field.register(serializers.FloatField)
 | |
| def convert_serializer_field_to_float(field):
 | |
|     return graphene.Float
 | |
| 
 | |
| 
 | |
| @get_graphene_type_from_serializer_field.register(serializers.DecimalField)
 | |
| def convert_serializer_field_to_decimal(field):
 | |
|     return graphene.Decimal
 | |
| 
 | |
| 
 | |
| @get_graphene_type_from_serializer_field.register(serializers.DateTimeField)
 | |
| def convert_serializer_field_to_datetime_time(field):
 | |
|     return graphene.types.datetime.DateTime
 | |
| 
 | |
| 
 | |
| @get_graphene_type_from_serializer_field.register(serializers.DateField)
 | |
| def convert_serializer_field_to_date_time(field):
 | |
|     return graphene.types.datetime.Date
 | |
| 
 | |
| 
 | |
| @get_graphene_type_from_serializer_field.register(serializers.TimeField)
 | |
| def convert_serializer_field_to_time(field):
 | |
|     return graphene.types.datetime.Time
 | |
| 
 | |
| 
 | |
| @get_graphene_type_from_serializer_field.register(serializers.ListField)
 | |
| def convert_serializer_field_to_list(field, is_input=True):
 | |
|     child_type = get_graphene_type_from_serializer_field(field.child)
 | |
|     return (graphene.List, child_type)
 | |
| 
 | |
| 
 | |
| @get_graphene_type_from_serializer_field.register(serializers.DictField)
 | |
| def convert_serializer_field_to_dict(field):
 | |
|     return DictType
 | |
| 
 | |
| 
 | |
| @get_graphene_type_from_serializer_field.register(serializers.JSONField)
 | |
| def convert_serializer_field_to_jsonstring(field):
 | |
|     return graphene.types.json.JSONString
 | |
| 
 | |
| 
 | |
| @get_graphene_type_from_serializer_field.register(serializers.MultipleChoiceField)
 | |
| def convert_serializer_field_to_list_of_enum(field):
 | |
|     child_type = convert_serializer_field_to_enum(field)
 | |
|     return (graphene.List, child_type)
 | |
| 
 | |
| 
 | |
| @get_graphene_type_from_serializer_field.register(serializers.ChoiceField)
 | |
| def convert_serializer_field_to_enum(field):
 | |
|     # enums require a name
 | |
|     name = field.field_name or field.source or "Choices"
 | |
|     return convert_choices_to_named_enum_with_descriptions(name, field.choices)
 |