3

我正在寻找这个常见问题的简单答案。到目前为止,我找到的答案为我们初学者留下了关键点。

我有一个应用程序,其中几乎每个模型都有一个外键到用户,并且有一个 unique_together 约束,其中一个字段始终是“用户”。

例如:

class SubscriberList(models.Model):

user = models.ForeignKey(User)
name = models.CharField(max_length=70)
date_created = models.DateTimeField(auto_now_add=True)

class Meta:
    unique_together = (
        ('user', 'name',),
        )   

def __unicode__(self):
    return self.name

订阅者列表总是由登录的用户创建,因此在创建订阅者列表的表单中,我排除了用户字段并在保存表单时给它一个 self.request.user 的值,如下所示:

class SubscriberListCreateView(AuthCreateView):
model = SubscriberList
template_name = "forms/app.html"
form_class = SubscriberListForm
success_url = "/app/lists/"

def form_valid(self, form):
    self.object = form.save(commit=False)
    self.object.user = self.request.user
    return super(SubscriberListCreateView, self).form_valid(form) 

这是随附的表格:

class SubscriberListForm(ModelForm):
class Meta:
    model = SubscriberList
    exclude = ('user')

使用此代码,有效数据就可以了。当我提交不是unique_together 的数据时,我从数据库中收到一个完整性错误。原因对我来说很清楚 - Django 不验证 unique_together 因为“用户”字段被排除在外。

如何更改我现有的代码,仍然使用 CreateView,以便提交的不是 unique_together 的数据会引发表单验证错误,而不是来自数据库的完整性错误。

4

3 回答 3

4

Yehonatan 的示例让我到达了那里,但我必须从 form_valid 的 ValidationError 中调用消息,而不是单独的 form_invalid 函数。

这有效:

class SubscriberCreateView(AuthCreateView):
    model = Subscriber
    template_name = "forms/app.html"
    form_class = SubscriberForm
    success_url = "/app/subscribers/"

    def form_valid(self, form):
        self.object = form.save(commit=False)
        self.object.user = self.request.user

        try:
            self.object.full_clean()
        except ValidationError:
            #raise ValidationError("No can do, you have used this name before!")
            #return self.form_invalid(form)
            from django.forms.util import ErrorList
            form._errors["email"] = ErrorList([u"You already have an email with that name man."])
            return super(SubscriberCreateView, self).form_invalid(form)

        return super(SubscriberCreateView, self).form_valid(form) 
于 2011-08-02T18:11:32.627 回答
1

从文档中获取: https ://docs.djangoproject.com/en/dev/ref/models/instances/?from=olddocs#validating-objects

如果您计划自己处理验证错误,或者如果您从 ModelForm 中排除了需要验证的字段,则只需要调用模型的 full_clean() 方法。

从文档中获取: https ://docs.djangoproject.com/en/dev/ref/class-based-views/#formmixin

混合 FormMixin 的视图必须提供 form_valid() 和 form_invalid() 的实现。

这意味着为了查看错误(与表单无关),您需要实现自己的 form_invalid,在此处添加特殊错误消息并将其返回。

因此,在您的对象上运行 full_clean() 应该会引发 unique_together 错误,因此您的代码可能如下所示:

def form_valid(self, form):
    self.object = form.save(commit=False)
    self.object.user = self.request.user
    # validate unique_together constraint
    try:
        self.object.full_clean()
    except ValidationError:
        # here you can return the same view with error messages
        # e.g.
        return self.form_invalid(form)
    return super(SubscriberListCreateView, self).form_valid(form)

def form_invalid(self, form):
    # using messages
    # from django.contrib import messages
    # messages.error('You already have a list with that name')
    # or adding a custom error
    from django.forms.util import ErrorList
    form._errors["name"] = ErrorList([u"You already have a list with that name"])
    return super(SubscriberListCreateView, self).form_invalid(form)

高温高压

于 2011-08-01T11:26:17.230 回答
0

添加另一个对新手来说可能更容易的示例。

表格.py

class GroupItemForm(ModelForm):
    def form_valid(self):
        self.object = self.save(commit=False)
        try:
            self.object.full_clean()
        except ValidationError:
            # here you can return the same view with error messages
            # e.g. field level error or...
            self._errors["sku"] = self.error_class([u"You already have an email with that name."])
            # ... form level error
            self.errors['__all__'] = self.error_class(["error msg"]
            return False
        return True

视图.py

def add_stock_item_detail(request, item_id, form_class=GroupItemForm, template_name="myapp/mytemplate.html"):
    item = get_object_or_404(Item, pk=item_id)
    product = Product(item=item)
    if request.method == 'POST':
        form = form_class(request.POST, instance=product)
        if form.is_valid() and form.form_valid():
            form.save()
            return HttpResponseRedirect('someurl')
    else:
        form = form_class(instance=product)

    ctx.update({
        "form" : form,
     })

    return render_to_response(template_name, RequestContext(request, ctx))
于 2011-10-05T12:35:23.860 回答