django-rest-framework中可写的嵌套序列化程序?

时间:2016-01-14 08:41:48

标签: python django rest

我的设计如下关于Django ModelSerializer。 有模型A和模型B.模型B有一个模型A的外键字段。由于某些原因,我不能直接使用主键来序列化模型B.我的想法,我需要的是序列化其他两个字段(在模型A)中独一无二。

我看到SlugRelatedField必须用于一个slug字段。 我搜索了一个NaturalKeyField可以支持NaturalKeyField。但看起来它被django-rest-framework所取代。但我检查了django-rest-framework,根本就没有这样的字段。 谁能帮忙?我该怎么办?

代码如下。 模型A

class AssetModel(models.Model):
    org = models.ForeignKey(Org, related_name='models')
    name = models.CharField(max_length=128)
    model_type = models.SmallIntegerField(default = 3,choices = MODEL_TYPE )
    directory = models.CharField(max_length = 128)
    ...
    class Meta:
        unique_together = ('org', 'name',)

模型B

class Dataitem(models.Model):
    mod = models.ForeignKey(AssetModel, related_name='dataitems')
    name = models.CharField(max_length=128)
    data_type = models.SmallIntegerField(default =0,choices = DATAITEM_DATATYPE)
    ...

模型A的串行器

class AssetModelSerializer(serializers.ModelSerializer):
    org =  serializers.SlugRelatedField(queryset=Org.objects.all(), slug_field='name')
    class Meta:
        model = AssetModel
        fields = ('org', 'name', 'model_type',..

B型串行器

class DataitemSerializer(serializers.ModelSerializer):
    class Meta:
        model = Dataitem
        fields = ('mod', 'name','data_type'...)

模型A的主键只是Django auto添加的id。序列化模型B时,我需要获取模型A的组织和名称。需要读取和写入。

2 个答案:

答案 0 :(得分:4)

嵌套序列化程序

您可以执行以下操作,为Dataitem定义可重用AssetModel模型的序列化程序

的序列化程序
class AssetModelSerializer(serializers.ModelSerializer):
    class Meta:
        model = AssetModel

    # Fields org and name of AssetModel will be inlcuded by default

class DataitemSerializer(serializers.ModelSerializer):
    class Meta:
        model = Dataitem

    mod = AssetModelSerializer()
        # This is the Dataitem.mod field
        # which is a FK to AssetModel,
        # Now it'll be serilized using the AssetModelSerializer
        # and include the org and name fields of AssetModelSerializer

我更喜欢这种方法,因为我得到了控制权。 如果使用上面的序列化,你会得到这样的结构:

data_item = {'name': ..., 'mod': {'org': ..., 'name': ...}}
                          ^
                          |___ AssetModel fields

或者您也可以使用depth = n

您还可以在depth = 1

中使用Dataitem
class DataitemSerializer(serializers.ModelSerializer):
        class Meta:
            model = Dataitem
            depth = 1 # Will include fields from related models
                      # e.g. the mod FK to AssetModel

Writable Nested Serializer

  

因为嵌套创建和更新的行为可能不明确,   并且可能需要相关模型之间的复杂依赖关系REST   框架3要求您始终明确地编写这些方法。

我们必须实施create/update以使DRF's documentation

成为可写的
class DataitemSerializer(serializers.ModelSerializer):
    class Meta:
        model = Dataitem

    # Nested serializer
    mod = AssetModelSerializer()

    # Custom create()
    def create(self, validated_data):
        # First we create 'mod' data for the AssetModel
        mod_data = validated_data.pop('mod')
        asset_model = AssetModel.objects.create(**mod_data)

        # Now we create the Dataitem and set the Dataitem.mod FK
        dataitem = Dataitem.objects.create(mod=asset_model, **validated_data)

        # Return a Dataitem instance
        return dataitem

答案 1 :(得分:2)

似乎有一个库可以做到这一点 drf-writable-nested

它处理这些类型的创建和序列化

  • OneToOne(直接/反向)
  • ForeignKey(直接/反向)
  • ManyToMany(直接/反向排除与通过模型的m2m关系)
  • GenericRelation(这始终只是反向)