【问题标题】:AttributeError at /api/updateorder/71 'RelatedManager' object has no attribute 'save'/api/updateorder/71 'RelatedManager' 对象的 AttributeError 没有属性 'save'
【发布时间】:2021-06-22 23:48:45
【问题描述】:

我正在尝试为订单对象制作更新 api。 OrderItem 模型与 Order 模型存在多对一关系(与外键)。此外,Billing Details 模型与 Order 模型是一对一的关系。我可以更新 Billing details 字段和 Order 字段,但无法更新 OrderItem 字段。

我收到此错误提示 'RelatedManager' 对象没有属性'save'

我的模型:

class Order(models.Model):
    ORDER_STATUS = (
        ('To_Ship', 'To Ship',),
        ('Shipped', 'Shipped',),
        ('Delivered', 'Delivered',),
        ('Cancelled', 'Cancelled',),
    )
    user = models.ForeignKey(User, on_delete=models.CASCADE, blank=True)    
    order_status = models.CharField(max_length=50,choices=ORDER_STATUS,default='To_Ship')

    ordered_date = models.DateTimeField(auto_now_add=True)
    ordered = models.BooleanField(default=False)
    total_price = models.CharField(max_length=50,blank=True,null=True)
    #billing_details = models.OneToOneField('BillingDetails',on_delete=models.CASCADE,null=True,blank=True,related_name="order")

    def __str__(self):
        return self.user.email

    class Meta:
        verbose_name_plural = "Orders"
        ordering = ('-id',)

class OrderItem(models.Model):
    #user = models.ForeignKey(User,on_delete=models.CASCADE, blank=True)
    order = models.ForeignKey(Order,on_delete=models.CASCADE, blank=True,null=True,related_name='order_items')
    item = models.ForeignKey(Product, on_delete=models.CASCADE,blank=True, null=True)
    order_variants = models.ForeignKey(Variants,on_delete=models.CASCADE,blank=True,null=True)
    quantity = models.IntegerField(default=1)
    # def price(self):
    #     total_item_price = self.quantity * self.item.varaints.price
    #     return total_item_price
    total_item_price = models.PositiveIntegerField(blank=True,null=True,)
    

    def __str__(self):
        return f"{self.quantity} items of {self.item} of {self.order.user}"

    class Meta:
        verbose_name_plural = "Cart Items"
        ordering = ('-id',)

class BillingDetails(models.Model):
    PAYMENT_TYPE = (
        ('cash_on_delivery', 'Cash On Delivery',),
        ('credit/debit_card', 'Credit/Debit Card',),
        ('connect_ips', 'Connect IPS',),
        ('fonepay', 'Fonepay',),
    )
    user = models.ForeignKey(User, on_delete=models.CASCADE, blank=True)
    order = models.OneToOneField(Order, on_delete=models.CASCADE, blank=True, null=True, related_name='billing_details')  
    address = models.CharField(max_length=100,blank=True,null=True)
    postal = models.CharField(max_length=50,blank=True,null=True)
    payment_type = models.CharField(max_length=50,blank=True,null=True,choices=PAYMENT_TYPE,default='cash_on-delivery')
    def __str__(self):
        return self.address

    class Meta:
        verbose_name_plural= "Shipping Address"

我的观点:

class UpdateOrderView(UpdateAPIView):
    permission_classes = [AllowAny]
    queryset = Order.objects.all()
    serializer_class = OrderUpdateSerializer

我的序列化器:

class OrderUpdateSerializer(serializers.ModelSerializer):
    order_items = OrderItemSerializer(many=True)
    billing_details = BillingDetailsSerializer()

    class Meta:
        model = Order
        fields = ['ordered','order_status','order_items','billing_details']


    def update(self, instance, validated_data):
        instance.order_status = validated_data.get('order_status')
        instance.ordered = validated_data.get('ordered')

        billing_details_data = validated_data.pop('billing_details',None)
        if billing_details_data is not None:
            instance.billing_details.address = billing_details_data['address']
            instance.billing_details.save() #no-error here


        order_items_data = validated_data.pop('order_items')
        for order_items_data in order_items_data:
            instance.order_items.quantity = order_items_data['quantity']
            instance.order_items.order_item_status = order_items_data['order_item_status']
            instance.order_items.save() #got the error here
            #super(OrderUpdateSerializer, self).update(instance.order_items,validated_data).save()
        #super().save()
        

        return super().update(instance,validated_data)

instance.billing_details.save()没有错误,但是同样的代码instance.order_items.save()会产生上述错误。如果我删除 instance.order_items.save(),那么它可以工作,但它不会将更新的数据保存在数据库中。

【问题讨论】:

  • instance.order_items 会给你一个相关的经理而不是OrderItem 的实例。
  • 是的,我放了 instance.order_items.save() 。那么,为什么会出现错误?
  • for order_items_data in order_items_data: 将不起作用,您在两个地方都使用相同的变量。这不是错误的直接原因,但它也会阻止您的代码按预期工作。

标签: django api django-rest-framework save postman


【解决方案1】:

您在一个模型中使用OneToOneField,但在另一个模型中使用ForeignKey,因此您不能对两者使用相同的代码。

class OrderItem(models.Model):
    ...
    order = models.ForeignKey(..., related_name='order_items')
    ...
 
class BillingDetails(models.Model):
    ...
    order = models.OneToOneField(..., related_name='billing_details')  
    ...

instance.billing_details 将返回一个BillingDetails 实例(如果有的话),因为它是一个OneToOneField,但instance.order_items 将返回一个与OrderItem 相关的管理器,而不是一个实例。

解决方案:

  1. 将字段OrderItem.order更改为OneToOneField;但我不知道这是否适合您的用例,因此您必须详细分析它是否适合您。

  2. instance.order_items.first()(或instance.order_items.get(...))从相关管理器获取实例。

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 2018-02-21
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2012-02-22
    • 1970-01-01
    • 2022-11-19
    • 1970-01-01
    相关资源
    最近更新 更多