python - Serializer 上的 SerializerClass 字段从主键保存

标签 python django serialization django-rest-framework django-serializer

我正在使用 Django-rest-framework 开发 API 并从网络应用程序使用它。它有一个带有来自 django.auth 用户模型的 Fk 的医生模型。我想从表单发布到医师模型,但序列化程序返回此消息:

{"user":{"non_field_errors":["Invalid data. Expected a dictionary, but got unicode."]}}

我正在发送用户对象的主键。在 DRF 上存储外键的正确方法(或只是一种方法)。我已尝试在序列化程序上覆盖 get_validation_exclusions 并在 View 集上覆盖 perform_create 方法。

api 和 web 应用程序是分离的。 API 是用 django 开发的,web 应用程序是用 angularjs 开发的。

我的模型

class Physician(models.Model):
    medical_office_number = models.CharField(max_length = 15)
    fiscal_id_number = models.CharField(max_length = 20)
    user = models.OneToOneField(User)

    def __unicode__(self):
        return self.user.first_name +' '+ self.user.last_name

序列化器:

class PhysicianSerializer(serializers.ModelSerializer):
    user = AccountSerializer()
    class Meta:
        model = Physician
        fields = ('id', 'user', 'medical_office_number', 'fiscal_id_number')
        read_only_fields = ('id')
        depth = 1
    def get_validation_exclusions(self, *args, **kwargs):
        exclusions = super(PhysicianSerializer, self).get_validation_exclusions()
        return exclusions + ['user']

*编辑 这是我的帐户序列化程序,它基于此实现并带有@Kevin Brown 的建议

class PrimaryKeyNestedMixin(serializers.RelatedField, serializers.ModelSerializer):

    def to_internal_value(self, data):
        return serializers.PrimaryKeyRelatedField.to_internal_value(self, data)
    def to_representation(self, data):
        return serializers.ModelSerializer.to_representation(self, data)

class AccountSerializer(PrimaryKeyNestedMixin):
    password = serializers.CharField(write_only=True, required=False)
    confirm_password = serializers.CharField(write_only=True, required=False)

    class Meta:
        model = Account
        fields = ('id', 'email', 'username', 'created_at', 'updated_at',
                  'first_name', 'last_name', 'password',
                  'confirm_password', 'is_admin',)
        read_only_fields = ('created_at', 'updated_at',)

View 集

class AccountViewSet(viewsets.ModelViewSet):
    lookup_field = 'username'
    queryset = Account.objects.all()
    serializer_class = AccountSerializer

当我尝试序列化这个对象时,它触发了一个错误。

所以我可以发布来自 <select> 的任何用户元素。但我无法验证解决方案。我缺少什么?

错误堆栈跟踪

TypeError at /api/v1/accounts/

__init__() takes exactly 1 argument (5 given)

Exception Location:     /home/jlromeroc/workspace/asclepios/venv/local/lib/python2.7/site-packages/rest_framework/relations.py in many_init, line 68
Python Executable:  /home/jlromeroc/workspace/asclepios/venv/bin/python
Python Version:     2.7.3

File "/home/jlromeroc/workspace/asclepios/venv/local/lib/python2.7/site-packages/django/core/handlers/base.py" in get_response 111. response = wrapped_callback(request, *callback_args, **callback_kwargs) File "/home/jlromeroc/workspace/asclepios/venv/local/lib/python2.7/site-packages/django/views/decorators/csrf.py" in wrapped_view 57. return view_func(*args, **kwargs)
File "/home/jlromeroc/workspace/asclepios/venv/local/lib/python2.7/site-packages/rest_framework/viewsets.py" in view 85. return self.dispatch(request, *args, **kwargs)
File "/home/jlromeroc/workspace/asclepios/venv/local/lib/python2.7/site-packages/rest_framework/views.py" in dispatch 407. response = self.handle_exception(exc) File "/home/jlromeroc/workspace/asclepios/venv/local/lib/python2.7/site-packages/rest_framework/views.py" in dispatch 404. response = handler(request, *args, **kwargs)
File "/home/jlromeroc/workspace/asclepios/venv/local/lib/python2.7/site-packages/rest_framework/mixins.py" in list 45. serializer = self.get_serializer(instance, many=True)
File "/home/jlromeroc/workspace/asclepios/venv/local/lib/python2.7/site-packages/rest_framework/generics.py" in get_serializer 90. instance, data=data, many=many, partial=partial, context=context File "/home/jlromeroc/workspace/asclepios/venv/local/lib/python2.7/site-packages/rest_framework/relations.py" in __new__ 48. return cls.many_init(*args, **kwargs)
File "/home/jlromeroc/workspace/asclepios/venv/local/lib/python2.7/site-packages/rest_framework/relations.py" in many_init 68. list_kwargs = {'child_relation': cls(*args, **kwargs)}

Exception Type: TypeError at /api/v1/accounts/
Exception Value: __init__() takes exactly 1 argument (5 given)

编辑** 我选择覆盖 View 集上的创建函数并将该对象包含在请求中,因此可以对其进行验证,但是随后,序列化程序会尝试为帐户模型插入一个新对象。我怎样才能防止这种行为?我试图将 PhysicianSerializer 类上的序列化程序设置为只读,但随后,django 尝试使用空 user_id 存储模型。如何在不尝试插入相关对象的情况下保存模型?

最佳答案

这里的问题是,对于嵌套序列化程序,Django REST 框架期望输入和输出都是嵌套表示。 DRF 将自动验证输入以确保它与嵌套序列化程序相匹配,从而允许您在单个请求中创建主对象和任何关系。

您正在寻找带有 PrimaryKeyRelatedField 输入的嵌套输出。这对于那些不需要在同一个请求中创建关系,而是总是在他们的关系中使用现有对象的人来说是很常见的。您必须采用的方法基本上是在 to_internal_value 中接收主键(就像 PrimaryKeyRelatedField),但在 to_representation< 中输出序列化程序。像这样的东西(未经测试)应该可以工作

class PrimaryKeyNestedMixin(serializers.PrimaryKeyRelatedField, serializers.ModelSerializer):

    def to_internal_value(self, data):
        return serializers.PrimaryKeyRelatedField.to_internal_value(self, data)

    def to_representation(self, data):
        return serializers.ModelSerializer.to_representation(self, data)

您需要将其用作嵌套序列化程序的混入,在您的情况下为 AccountSerializer,它应该做您正在寻找的事情。

关于python - Serializer 上的 SerializerClass 字段从主键保存,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28010663/

相关文章:

python - 只能将 '.sparse' 访问器与稀疏数据一起使用

python - Django不一致迁移历史: Migration X is applied before its dependency Y on database 'default'

javascript - 在 Django 中将字符串转换为 Javascript 中的字典列表

json - 如何从列表中映射 Flutter JSON 字符串?

java - 持久化的序列化缺点

python - OCR手写数据显示svm.train()中的错误

python - Python 2.2.3 中的 SQLite

python - 使用 Pandas 拆分数据

python - 改进模板标签的功能 - Django

java - 缓存大对象并在需要时反序列化它们 (Java)