REST 中的多对多字段和嵌套序列化器:覆盖嵌套序列化器不会在 Django 中创建嵌套对象

Mic*_*gas 2 django django-models django-serializer django-rest-framework

我有两个具有多对多关系的模型,我正在尝试将嵌套数据发送到我的 API。不幸的是它只给我返回一个空数组。

这就是我正在尝试的:

我的模型:


class Building(models.Model):
    name  = models.CharField(max_length=120, null=True, blank=True)
    net_leased_area = models.FloatField(null=True, blank=True)

class BuildingGroup(models.Model):
    description           = models.CharField(max_length=500, null=True, blank=True)
    buildings             = models.ManyToManyField(Building, default=None, blank=True)

Run Code Online (Sandbox Code Playgroud)

我的通用 API 视图:

class BuildingGroupCreateAPIView(CreateAPIView):
    queryset                    = BuildingGroup.objects.all()
    serializer_class            = BuildingGroupSerializer
Run Code Online (Sandbox Code Playgroud)

我的序列化器:


class BuildingGroupSerializer(serializers.ModelSerializer):

    buildings = BuildingSerializer(many=True)

    class Meta:

        model = BuildingGroup

        fields = (
            'description',
            'buildings',
        )

    def create(self, validated_data):
        buildings_data = validated_data.pop('buildings')
        building_group = BuildingGroup.objects.create(**validated_data)
        for building_data in buildings_data:
            Building.objects.create(building_group=building_group, **building_data)
        return building_group

Run Code Online (Sandbox Code Playgroud)

当我发送数据时它返回:


{"description":"Test Description API","buildings":[]}

Run Code Online (Sandbox Code Playgroud)

在数组中我想要有我的字典数组。

当我覆盖 create 方法以发送嵌套对象时,我尝试遵循此处的 REST 文档。(https://www.django-rest-framework.org/api-guide/relations/#writable-nested-serializers)我认为我做得正确,但是史诗般的失败。

我使用自定义方法发送请求数据,如下所示:

test_api_local(method="post", data={
        "description": "Test Description API",
        "buildings": [{'name' : 'Testname'}, .... ],
         })
Run Code Online (Sandbox Code Playgroud)

非常感谢任何帮助。非常感谢!!

编辑:当我尝试在 REST 视图上测试它时,它告诉我:

TypeError: 'building_group' is an invalid keyword argument for this function

EDIT2:这是我的观点:

class BuildingGroupCreateAPIView(CreateAPIView):
    queryset                    = BuildingGroup.objects.all()
    serializer_class            = BuildingGroupSerializer

    def create(self, request, *args, **kwargs):
        serializer = BuildingGroupSerializer(data=self.request.data)
        serializer.is_valid(raise_exception=True)
        serializer.save()
        return Response(serializer.data)
Run Code Online (Sandbox Code Playgroud)

Aks*_*ngh 5

您必须Building根据有效负载数据内传递的 id 显式获取或创建实例,然后将它们添加到BuildingGroup实例中。

class NestedBuildingSerializer(serializers.ModelSerializer):
    id = serializers.IntegerField(required=False)

    class Meta:
        model = Building
        fields = '__all__'


class BuildingGroupSerializer(serializers.ModelSerializer):
    buildings = NestedBuildingSerializer(many=True)

    class Meta:
        model = BuildingGroup
        fields = (
            'description',
            'buildings',
        )

    def create(self, validated_data):
        buildings_data = validated_data.pop('buildings')
        building_group = BuildingGroup.objects.create(**validated_data)
        buildings = []  # it will contains list of Building model instance
        for building_data in buildings_data:
            building_id = building_data.pop('id', None)
            building, _ = Building.objects.get_or_create(id=building_id,
                                                         defaults=building_data)
            buildings.append(building)
        # add all passed instances of Building model to BuildingGroup instance
        building_group.buildings.add(*buildings)
        return building_group

class BuildingGroupView(ListAPIView, CreateAPIView):
    queryset = BuildingGroup.objects.all()
    serializer_class = BuildingGroupSerializer


## Assume you add your views like this in urls.py
urlpatterns = [
    .....
    path('building-groups', views.BuildingGroupView.as_view(),
         name='building-group'),
    .....
]
Run Code Online (Sandbox Code Playgroud)

在使用如下有效负载/building-groups作为POST方法调用端点时:

{
  "description": "here description",
  "buildings": [
    {
      "id": 1, # existing building of id 1
      "name": "name of building 1",
      "net_leased_area": 1800
    },
    { 
      # a new building will gets create
      "name": "name of building 2",
      "net_leased_area": 1800
    }
  ]
}
Run Code Online (Sandbox Code Playgroud)

然后,它将返回如下响应:-

{
  "description": "here description",
  "buildings": [
    {
      "id": 1,
      "name": "name of building 1",
      "net_leased_area": 1800
    },
    {
      "id": 2
      "name": "name of building 2",
      "net_leased_area": 1800
    }
  ]
}
Run Code Online (Sandbox Code Playgroud)

了解M2M 关系 ,并且,.get_or_create()

注意:BuildingSerializerNestedBuildingSerializer都是不同的。不要把它们搞混了。