使用表单集前缀时缺少 ManagementForm 数据

ManagementForm Data is Missing When Using Formset Prefix

有人知道为什么当我使用 Formset 前缀时会引发 ManagementForm Data is Missing 吗?

来自 Shell

>>> from django import forms
>>> from django.forms.formsets import formset_factory
>>> 
>>> class CheckBox (forms.Form):
...     overwrite = forms.BooleanField (required = False)
... 
>>> 
>>> data = {
...     'form-TOTAL_FORMS': '2',
...     'form-INITIAL_FORMS': '0',
...     'form-MAX_NUM_FORMS': '3',
...     'checkbox-0-overwrite': True,
...     'checkbox-1-overwrite': False,
... }
>>> 
>>> CheckBoxFormSet = formset_factory (CheckBox)
>>> formset = CheckBoxFormSet (data)
>>> formset.is_valid ()
True
>>> formset.cleaned_data
[{}, {}]
>>> 

为 Formset 添加前缀

>>> formset = CheckBoxFormSet (data, prefix = 'checkbox')
>>> formset.is_valid ()
Traceback (most recent call last):
  File "<console>", line 1, in <module>
.
.
.
django.core.exceptions.ValidationError: ['ManagementForm data is missing or has been tampered with']

Django Doc 提到在 'a' 视图中使用前缀来区分不同的表单集。如果我在同一个视图中使用它,但使用不同的方法处理不同的 HTML 页面(如示例),这是否适用?执行示例中 Django 的建议也会触发 ManagementForm Data is Missing Error。

例如:

forms.py

class NodeForm (forms.Form):

    cars = forms.CharField (required = False)
    trucks = forms.CharField (required = False)

class CheckBox (forms.Form):
    overwrite = forms.BooleanField (required = False)

views.py

def cars (request):

    CarsFormSet = formset_factory (CarsForm, formset = BaseNodeFormSet, extra = 2, max_num = 5)

    if request.method == 'POST':

        cars_formset = CarsFormSet (request.POST, prefix = 'carsform')

        if cars_formset.is_valid ():
            data = cars_formset.cleaned_data

            context = {'data': data}
            return render (request, 'vehicleform/response.html', context)
        else:
            cars_formset = CarsFormSet (prefix = 'carsform')

     context = {...previously entered data from POST...}

     return render (request, 'vehicleform/carsform.html', context)

def trucks (request):

    TrucksFormSet = formset_factory (TrucksForm, extra = 2, max_num = 5)

    if request.method == 'POST':

        trucks_formset = TrucksFormSet (request.POST, prefix = 'trucksform')

        if trucks_formset.is_valid ():
            data = truck_formset.cleaned_data

            context = {'data': data}
            return render (request, 'vehicleform/success.html', context)

        else:
            trucks_formset = TrucksFormSet (prefix = 'trucksform')

     return HttpResponse ('No overwrite data.')

更新 1
我已将其缩小到实际数据。它出于某种原因不喜欢我的数据。

更新 2
我已经验证了表单中的名称和数据是相同的。它只打印一个 checkbox-0-overwrite,而我在我的数据中声明了 2。想知道为什么 formset 对复选框不起作用。

>>> CheckBoxFormSet = formset_factory (CheckBox)
>>> formset = CheckBoxFormSet (prefix = 'checkbox')
>>> 
>>> for form in formset:
...     print (form)
... 
<tr><th><label for="id_checkbox-0-overwrite">Overwrite:</label></th><td><input id="id_checkbox-0-overwrite" name="checkbox-0-overwrite" type="checkbox" /></td></tr>
>>> 

更新 3
我不确定发生了什么。这似乎生成了没有前缀的表单。我插入前缀时仍然出现错误。

>>> CheckBoxFormSet = formset_factory (CheckBox)
>>> formset = CheckBoxFormSet (data)
>>> formset.is_valid ()
True
>>> for form in formset:
...     print (form)
... 
<tr><th><label for="id_form-0-overwrite">Overwrite:</label></th><td><input id="id_form-0-overwrite" name="form-0-overwrite" type="checkbox" /></td></tr>
<tr><th><label for="id_form-1-overwrite">Overwrite:</label></th><td><input id="id_form-1-overwrite" name="form-1-overwrite" type="checkbox" /></td></tr>
>>> 
>>> 
>>> data {
...    'form-TOTAL_FORMS': '2',
...    'form-INITIAL_FORMS': '0',
...    'form-MAX_NUM_FORMS': '3',
...    'checkbox-0-overwrite': True
}

更新 4
下面的 html 模板是由第一个表单 cars 生成和创建的,正如我从上面的示例中更新的那样。第二种形式只在第一种形式传递的数据旁边插入复选框。在模板中显示表单集并单击提交仍然给我 "ManagementForm" 错误。我将尝试创建一个只有复选框的全新表单,看看是否会给我任何错误。

Response.html

<!DOCTYPE html>
<html xmlns="http://www.w3.org/1999/html">
<head lang="en">
    <meta charset="UTF-8">
    {% load staticfiles %}
    <link rel="stylesheet" type="text/css" href="{% static 'nodeform/style.css' %}" >
    <title>Vehicle Information</title>
</head>
<body>
    <h1>Vehicle Information:</h1>
    <h4>Location: {{ location }}</h4>
    <form action="trucks" method="POST">{% csrf_token %}
    {{ checkbox_formset.management_form }}
       {% for form in checkbox_formset %}
        {{ form }}
       {% endfor %}
    <br>
    <p><input type="submit" value="Confirm">
    <a href="{% url 'carsform' %}">
        <button type="button">Cancel</button></a></p>
    </form>
</body>
</html>

更新 5
我不确定我是否理解正确,但我认为失败在于表单的操作以及我如何获取数据。初始表单 (carsform.html) 的表单标签没有任何操作:

carsform.html

<form action="" method="POST">{% csrf_token %}...</form>

它执行一个POST,然后将收集到的信息传递给下一个page/form(response.html)。此外,它向以前的数据添加了一组复选框,如下所示:

response.html

<form action="trucks" method="POST">{% csrf_token %}...</form>

输出:

Audi (Obtained from cars)    []  <---Checkbox inserted from response.html manually & obtaining data from method trucks
Toyota (Obtained from cars)  []  <---Checkbox inserted from response.html manually & obtaining data from method trucks

当用户点击 "Submit" 时,response.html 表单将进行处理,然后 "reverse" 再次返回卡车。这次没有来自 cars 方法的数据要处理。这最终引发了 ManagementForm 错误。

我已经通过在初始页面 (carsform.html) 中插入 2 个表单集并单击提交来对此进行了测试。我在下一个 page/form (response.html) 上看到的结果同时具有第一个和第二个 formset 的数据。

我的下一个问题是如何创建第二个表单(response.html)来获取数据而不会出错?

根据 docs:

It is important to point out that you need to pass prefix on both the POST and non-POST cases so that it is rendered and processed correctly.

因此,首先,在呈现空白表单时(即不是 POST),您将拥有:

trucks_formset = TrucksFormSet(prefix ='trucksform')

您的 'data' 也因前缀失败而失败,因为您没有更改数据中字段的名称。前缀重命名您的字段。您可以尝试将表单集发布到模板,您将看到隐藏字段的名称。

 data = {
...     'checkbox-TOTAL_FORMS': '2',
...     'checkbox-INITIAL_FORMS': '0',
...     'checkbox-MAX_NUM_FORMS': '',
...     'checkbox-0-overwrite': True,
...     'checkbox-1-overwrite': False,
... }
>>> 
>>> CheckBoxFormSet = formset_factory (CheckBox, extra=1)
>>> formset = CheckBoxFormSet (data, prefix = 'checkbox')

问题在于方法 cars 呈现到 response.html 页面并在 url http://..../vehicle/cars 而不是 vehicle/trucks 处显示呈现的表单。引发 "Management Form Error" 是因为 "POST" 发生了第二次,同时仍处于 url vehicle/cars 形式而不是 vehicle/trucks 形式。更新 5 提示了问题。解决方案是简单地使用

return HttpResponseRedirect ('trucks')

render (request, 'vehicleform/trucksform.html', context)
return HttpResponseRedirect ('trucksform')

上述两者的区别在于第一个解决方案呈现来自第二种形式 (trucksform) 的数据,而第二种解决方案呈现来自第一种形式 (carsform) 的数据。

为什么这么重要?好吧,因为我希望第一个表单在出现错误时重新显示自己而不重定向到另一个页面;因此,

<form action="" method="POST">

否则,设置

<form action="truck" method="POST">

不会造成这种混乱。

为了能够在一个视图中使用 2 个不同的表单集,请转到它们的直接 URL 分别测试每个 page/form。一旦确认两个页面都按预期呈现和工作,请使用 HttpResponseRedirect。

感谢 onyeka 全程帮助。

这不完全是 OP 的问题,但这是我搜索此错误时出现的第一个结果,所以我想我会分享我的前缀(包含在内)的问题,因为它不是显而易见。

Django 表单框架 automatically defines _id_for_label 如果您不为表单字段的属性设置一个,但不从表单集数据中删除它们。因此,如果您的标签是 "my_label",它将呈现为 "id_my_label",并返回 POST 数据。

我花了一段时间才在我的 request.POST 中发现这个(下面的 data

MyFormSet = formset_factory(MyForm, prefix='my_form')

data = {
    'id_result_form-MAX_NUM_FORMS': '1000', # note how django added 'id_'
    'id_result_form-INITIAL_FORMS': '0',
    'id_result_form-TOTAL_FORMS': '1',
}

formset = MyFormSet(data, prefix='my_form')
formset.is_valid()
>>> False

formset = MyFormSet(data, prefix='id_my_form') # added 'id_'
formset.is_valid()
>>> True