python - 保持 Django 模型清洁方法验证外键对象并使用 ModelForm 保存

标签 python django django-models django-forms django-validation

在我的模型的 clean 方法中,我验证外键 exhibitor is_premium 是否给出,并且验证他没有超过 MAX_DISCOUNTS_PER_EXHIBITOR 事件对象。

它在 django admin 中工作得很好。添加时以及编辑时。

我想让它在我的自定义(非 django-admin) View 中工作。我正在使用模型表单。

第一种方法

我将 exhibitor in view 分配给最初 commit=False 保存的对象。

它因 DoesNotExist 而崩溃:折扣没有参展商,因为执行了 clean,但尚未分配参展商。正确的实现方式应该是什么?

模型.py

class Discount(models.Model):
    exhibitor = models.ForeignKey(
        'core_backend.Exhibitor', related_name='discounts')
    is_active = models.BooleanField(default=False, verbose_name=u"Aktywny")
    title = models.TextField(verbose_name=u"Tytuł")

    def clean(self):

        if not self.exhibitor.is_premium:
            raise ValidationError(
                u'Discounts only for premium')

        count = Discount.objects.filter(
            is_active=True,
            exhibitor=self.exhibitor
        ).count()

        if not self.pk:
            # newly added
            count = count + (1 if self.is_active else 0)
        else:
            # edited
            discount = Discount.objects.get(pk=self.pk)
            if discount.is_active and not self.is_active:
                count = count - 1
            elif not discount.is_active and self.is_active:
                count = count + 1

        if count > settings.MAX_DISCOUNTS_PER_EXHIBITOR:
            raise ValidationError(
                u'Max %s active discounts' % (
                    settings.MAX_DISCOUNTS_PER_EXHIBITOR
                )
            )

表格.py

class DiscountForm(forms.ModelForm):

    class Meta:
        model = Discount
        fields = (
            'title',
            'description',
            'is_activa',
        )

View .py

def add_discount(request, fair_pk, exhibitor_pk):
    fair = get_object_or_404(Fair, pk=fair_pk)
    exhibitor = get_object_or_404(
        Exhibitor, fair=fair, pk=exhibitor_pk, user=request.user)

    if request.method == 'GET':

        form = DiscountForm()

        return render(request, 'new_panel/add_discount.html', {
            'exhibitor': exhibitor,
            'discount_form': form,
        })

    if request.method == 'POST':
        form = DiscountForm(data=request.POST)

        if form.is_valid():
            discount = form.save(commit=False)
            discount.exhibitor = exhibitor
            discount.save()
            return redirect(reverse(
                'discounts_list',
                kwargs={"fair_pk": fair.pk, 'exhibitor_pk': exhibitor.pk}
            ))
        return render(request, 'new_panel/add_discount.html', {
            'exhibitor': exhibitor,
            'discount_form': form,
        })

第二种方法

我也尝试了不同的方法,使用单独的 POST 表单,但它因同样的错误而崩溃:

表格.py

class DiscountFormAdd(forms.ModelForm):

    class Meta:
        model = Discount
        fields = (
            'title',
            'exhibitor',
            'is_active',
        )

    def __init__(self, exhibitor, *args, **kwargs):
        self.exhibitor = exhibitor
        super(DiscountFormAdd, self).__init__(*args, **kwargs)

    def save(self, commit=False):
        discount = super(DiscountFormAdd, self).save(commit=False)
        discount.exhibitor = self.exhibitor

        if commit:
            discount.save()

        return discount

View .py

def add_discount(request, fair_pk, exhibitor_pk):
    fair = get_object_or_404(Fair, pk=fair_pk)
    exhibitor = get_object_or_404(
        Exhibitor, fair=fair, pk=exhibitor_pk, user=request.user)

    if request.method == 'GET':

        form = DiscountForm()

        return render(request, 'new_panel/add_discount.html', {
            'exhibitor': exhibitor,
            'discount_form': form,
        })

    if request.method == 'POST':
        form = DiscountFormAdd(data=request.POST, exhibitor=exhibitor)

        if form.is_valid():
            discount = form.save(commit=True)

            return redirect(reverse(
                'discounts_list',
                kwargs={"fair_pk": fair.pk, 'exhibitor_pk': exhibitor.pk}
            ))
        return render(request, 'new_panel/add_discount.html', {
            'exhibitor': exhibitor,
            'discount_form': form,
        })

我想坚持模型清理验证,而不是跳入表单清理。如果 django admin 能够做到这一点,那么肯定可以在自定义模型中实现。

最佳答案

当使用 self.exhibitor = exhibitor 分配参展商时,这会在您的表单上设置一个与表单内容无关的属性

实际设置参展商,请改用以下代码:

import copy
.
.
.
    if request.method == 'POST':
        # request.POST is an immutable QueryDict so it needs to be copied
        form_data = copy.copy(request.POST)
        form_data['exhibitor'] = exhibitor.id
        form = DiscountFormAdd(data=form_data)

并完全放弃在表单的 __init__

中设置 exhibitor

这样,您的表单数据(用于创建您的折扣)对象将是正确的并且可以被清除

[编辑] 如何在 ModelForm 中创建隐藏的输入字段:

class DiscountFormAdd(forms.ModelForm):
    def __init__(self, *args, **kwargs):
        super(DiscountFormAdd, self).__init__(*args, **kwargs)
        self.fields['exhibitor'].widget = forms.HiddenInput()

    ...

或者你也可以使用 Meta 类的 widgets 属性

class DiscountFormAdd(forms.ModelForm):

    class Meta:
        model = Discount
        widgets = {'exhibitor': forms.HiddenInput()}
        ...

关于python - 保持 Django 模型清洁方法验证外键对象并使用 ModelForm 保存,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29644269/

相关文章:

python - 根据输入对大列表进行切片

django - 在 Django Rest Framework 中使用 HyperlinkedField 进行序列化器

python - 类型错误 : __init__() got an unexpected keyword argument 'decimal_places'

django - 在另一个应用程序模型中导入应用程序模型类

python - 为什么我不能访问 Meta 类作为 Django 模型类的属性?

python - 如何在 Tkinter 中的命令中触发按钮的更改?

python - 在 Twisted Python 中——确保协议(protocol)实例将被完全释放

python - UnboundLocalError : local variable . ..赋值前引用

python - 如何处理 django 模型中的模型通用属性?

Django 标记 : "already registered" exception