如何在django rest框架中保存嵌套关系?

时间:2016-02-18 12:00:14

标签: python django python-2.7 django-rest-framework

我试图在Django REST Framework guide之后在Django REST Framework中保存嵌套关系,但我无法做到。

我有一个班级“Asiento”,这是另一个班级“Apunte”的外键。当我得到每个“Asiento”时,Django REST Framework将返回它们以及所有“Apunte”对象。这是有效的,但当我尝试使用可写嵌套序列化器创建/更新“Asiento”时,我得到AsientoSerializer(data=data).is_valid() == False

我的模特:

class Apunte(TxerpadBase):
    debe = models.DecimalField(null=True, blank=True, max_digits=18, decimal_places=6)
    haber = models.DecimalField(null=True, blank=True, max_digits=18, decimal_places=6)
    cuenta = models.ForeignKey(Cuenta, related_name='mayor', on_delete=models.PROTECT)
    partner = models.ForeignKey(Partner, null=True, blank=True, on_delete=models.PROTECT)
    asiento = models.ForeignKey(Asiento, related_name='apuntes')
    fecha = models.DateField()
    recc = models.BooleanField(blank=True, default=False)
    conciliacion = models.ForeignKey('Conciliacion', null=True, blank=True, on_delete=models.SET_NULL)
    estado = FSMField(default='borrador')


class Asiento(TxerpadBase):
    numero = models.PositiveIntegerField(null=True, blank=True)
    fecha = models.DateField(blank=True, default=datetime.datetime.now)
    libro = models.ForeignKey('Libro', on_delete=models.PROTECT)
    periodo = models.ForeignKey(Periodo, on_delete=models.PROTECT)
    estado = FSMField(default='borrador')

我的序列化器:

class ApunteSerializer(serializers.ModelSerializer):
    fecha = serializers.DateField(
        format='%d-%m-%Y', input_formats=('%d-%m-%Y',),
        error_messages={'invalid': 'La fecha del apunte no esta en el formato correcto.'}
    )

    class Meta:
        model = Apunte


class AsientoSerializer(serializers.ModelSerializer):
    fecha = serializers.DateField(
        format='%d-%m-%Y', input_formats=('%d-%m-%Y',),
        error_messages={'invalid': 'La fecha del asiento no esta en el formato correcto.'}
    )
    apuntes = ApunteSerializer(many=True)

    class Meta:
        model = Asiento

    def create(self, data):
        apuntes_data = data['apuntes']
        asiento_data = data
        asiento_data['fecha'] = datetime.datetime.strptime(asiento_data['fecha'], '%d-%m-%Y').date()
        del asiento_data['apuntes']
        asiento = Asiento.objects.create(**asiento_data)
        for apunte in apuntes_data:
            apunte['fecha'] = datetime.datetime.strptime(apunte['fecha'], '%d-%m-%Y').date()
            Apunte.objects.create(asiento=asiento, **apunte)

我的观点:

class AsientoViewSet(viewsets.ModelViewSet):
    queryset = Asiento.objects.all()
    serializer_class = AsientoSerializer

    def create(self, validated_data):
        # JSON dictionary is inside validated_data.data
        serializer = AsientoSerializer(data=validated_data.data)
        if serializer.is_valid():
            return Response(serializer.data)
        else:
            raise AttributeError('Error al validar')

这是我随请求发送的JSON:

{
    u'name': u'prueba 3000', 
    u'periodo': 13, 
    u'fecha': u'18-02-2016', 
    u'numero': None, 
    u'estado': u'borrador', 
    u'libro': 1, 
    u'apuntes': [
        {
            u'name': u'a', 
            u'recc': False, 
            u'debe': u'1', 
            u'haber': u'0.00', 
            u'cuenta': u'5', 
            u'partner': 8, u'fecha': 
            u'18-02-2016', 
            u'conciliacion': u''
        }
    ]
}

如果我只调试代码,那么执行的“create”方法是viewset的方法,但它不会从序列化程序运行“create”方法。

使用此代码,我无法保存“Asiento”既不是“Apunte”。我究竟做错了什么?谢谢你的回复!

我使用的是python 2.7(由于外部原因我无法更新)和Django REST Framework 3.3.2。

3 个答案:

答案 0 :(得分:1)

好的,我发现我的问题是" asiento"属于" Apunte"对象不能为空,我也没有发送它。现在,我已经改变了#asiento"属于" Apunte"对象和序列化器终于可以工作,而且不需要"创建" ViewSet上的方法。

感谢大家的回答。

答案 1 :(得分:0)

在序列化无效的情况下,您应该首先查看返回的消息(serializer.data)。它可以帮助您了解出了什么问题。

我的盲目猜测是你错过了与嵌套序列化程序相关的queryset参数。如果未设置,DRF会将这些字段视为只读。

编辑:在OP版本之后,问题来自视图集。 如果序列化器有效,则需要调用serializer.save()。 看看CreateMixin是如何做到的。

答案 2 :(得分:0)

不确定这是否是解决方案。但是序列化程序的create方法必须返回创建的对象。 尝试将return添加到您的上一行