python - clean_data 在 Django 1.11 中消失在哪里?

标签 python django formset inline-formset

我创建了一个 inlineformset_factory 如下:

formset = inlineformset_factory(Author, Book, form=BookForm,
                                formset=BaseBookFormSet,
                                can_order=False, can_delete=True,
                                extra=1, fields=('id', name)
                                )

BookForm如下:

class BookForm(forms.ModelForm):
    name = forms.Charfield(required=True)

    def __init__(self, *args, **kwargs):
        super(BookForm, self).__init__(*args, **kwargs)
        self.helper = FormHelper()
        self.helper.form_tag = False
        self.helper.layout = Layout(
            Div(
                Field("id", type="hidden"),
                Field("name"),
                Field("DELETE")
                )
    )

    class Meta:
        model = Book
        fields = ('id', 'name')

    def clean_name(self):
        book_name = self.cleaned_data['name']
        try:
            book = Book.objects.get(name=book_name)
            return book
         except:
            return book_name

    def clean(self):
        cleaned_data = super(BookForm, self).clean()
        ... other operations on cleaned_data ...

    def has_changed(self):
        changed = super(BookForm, self).has_changed()
        cleaned_data = self.clean()
        ... other code here ...

提交表单时抛出错误:

Exception Type: AttributeError
Exception Value: 'BookForm' object has no attribute 'cleaned_data'

当在views.py中调用formset.is_valid()时。 Traceback 首先显示 has_changed 中调用 self.clean 的行,然后显示 clean() 中调用 super clean 的行。

这在 django 1.10 中运行良好。

当我尝试在 Django 1.10 中打印 dir(self) 时,它确实将“cleaned_data”显示为属性之一,而在 Django 1.11 中却没有。

Django 1.11 中的“cleaned_data”消失在哪里?

编辑:添加回溯:

Traceback (most recent call last):
  File "/home/vagrant/venv/local/lib/python2.7/site-packages/django/core/handlers/exception.py", line 41, in inner
    response = get_response(request)
  File "/home/vagrant/venv/local/lib/python2.7/site-packages/django/core/handlers/base.py", line 249, in _legacy_get_response
    response = self._get_response(request)
  File "/home/vagrant/venv/local/lib/python2.7/site-packages/django/core/handlers/base.py", line 187, in _get_response
    response = self.process_exception_by_middleware(e, request)
  File "/home/vagrant/venv/local/lib/python2.7/site-packages/django/core/handlers/base.py", line 185, in _get_response
    response = wrapped_callback(request, *callback_args, **callback_kwargs)
  File "/home/vagrant/venv/local/lib/python2.7/site-packages/django/views/generic/base.py", line 68, in view
    return self.dispatch(request, *args, **kwargs)
  File "/home/vagrant/venv/local/lib/python2.7/site-packages/django/views/generic/base.py", line 88, in dispatch
    return handler(request, *args, **kwargs)
  File "/vagrant/test_os/inventory/views.py", line 297, in post
    if formset.is_valid():
  File "/home/vagrant/venv/local/lib/python2.7/site-packages/django/forms/formsets.py", line 321, in is_valid
    self.errors
  File "/home/vagrant/venv/local/lib/python2.7/site-packages/django/forms/formsets.py", line 295, in errors
    self.full_clean()
  File "/home/vagrant/venv/local/lib/python2.7/site-packages/django/forms/formsets.py", line 345, in full_clean
    if not form.has_changed():
  File "/vagrant/test_os/inventory/forms.py", line 220, in has_changed
    cleaned_data = self.clean()
  File "/vagrant/test_os/inventory/forms.py", line 177, in clean
    cleaned_data = super(BookForm, self).clean()
  File "/home/vagrant/venv/local/lib/python2.7/site-packages/django/forms/models.py", line 344, in clean
    return self.cleaned_data
AttributeError: 'BookForm' object has no attribute 'cleaned_data'

最佳答案

表单集在 1.11 ( in #26844 ) 中进行了修复,以在验证最小数量的表单时忽略空表单。作为副作用,表单集现在在验证表单之前对每个表单调用 form.has_changed()。 Django 希望在验证表单之前可以安全地调用 form.has_changed(),并且默认实现确实可以安全地调用。

您已经重写了 form.has_changed() 以调用 self.clean(),这现在发生在验证表单之前。由于 form.clean() 要求验证表单,因此会失败。

由于 form.full_clean() 实际上调用 self.has_changed(),因此您不能简单地从 form.has_changed()< 内部验证表单。您没有显示您在 has_changed() 中所做的事情,但将此代码放在其他地方很可能是个好主意。

关于python - clean_data 在 Django 1.11 中消失在哪里?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/43320603/

相关文章:

c# - python:math.sqrt(x) 函数有多准确?

Django 1.8 - migrate 和 makemigrations 之间有什么区别?

django - this.$apollo 始终未定义

Django 模型表单集显示数据库数据而不是空表单集

python - 表单集中所需的第一个表单

python - 从半径为 R1 和 R2 的圆环内的图像中提取数据点

python - 如何使用列表快速更新大字典?

django - 将自定义表单参数传递给表单集

Python-局部变量保留重复调用函数的内容

python - 在 Django 中定义常量