Django formset unit test
Я не могу запустить Unit Test с помощью набора форм.
Я пытаюсь выполнить тест:
class NewClientTestCase(TestCase):
def setUp(self):
self.c = Client()
def test_0_create_individual_with_same_adress(self):
post_data = {
'ctype': User.CONTACT_INDIVIDUAL,
'username': 'dupond.f',
'email': '[email protected]',
'password': 'pwd',
'password2': 'pwd',
'civility': User.CIVILITY_MISTER,
'first_name': 'François',
'last_name': 'DUPOND',
'phone': '+33 1 34 12 52 30',
'gsm': '+33 6 34 12 52 30',
'fax': '+33 1 34 12 52 30',
'form-0-address1': '33 avenue Gambetta',
'form-0-address2': 'apt 50',
'form-0-zip_code': '75020',
'form-0-city': 'Paris',
'form-0-country': 'FRA',
'same_for_billing': True,
}
response = self.c.post(reverse('client:full_account'), post_data, follow=True)
self.assertRedirects(response, '%s?created=1' % reverse('client:dashboard'))
и у меня есть эта ошибка:
ValidationError: данные [u'ManagementForm отсутствуют или были подделано ']
Мой взгляд:
def full_account(request, url_redirect=''):
from forms import NewUserFullForm, AddressForm, BaseArticleFormSet
fields_required = []
fields_notrequired = []
AddressFormSet = formset_factory(AddressForm, extra=2, formset=BaseArticleFormSet)
if request.method == 'POST':
form = NewUserFullForm(request.POST)
objforms = AddressFormSet(request.POST)
if objforms.is_valid() and form.is_valid():
user = form.save()
address = objforms.forms[0].save()
if url_redirect=='':
url_redirect = '%s?created=1' % reverse('client:dashboard')
logon(request, form.instance)
return HttpResponseRedirect(url_redirect)
else:
form = NewUserFullForm()
objforms = AddressFormSet()
return direct_to_template(request, 'clients/full_account.html', {
'form':form,
'formset': objforms,
'tld_fr':False,
})
и мой файл формы:
class BaseArticleFormSet(BaseFormSet):
def clean(self):
msg_err = _('Ce champ est obligatoire.')
non_errors = True
if 'same_for_billing' in self.data and self.data['same_for_billing'] == 'on':
same_for_billing = True
else:
same_for_billing = False
for i in [0, 1]:
form = self.forms[i]
for field in form.fields:
name_field = 'form-%d-%s' % (i, field )
value_field = self.data[name_field].strip()
if i == 0 and self.forms[0].fields[field].required and value_field =='':
form.errors[field] = msg_err
non_errors = False
elif i == 1 and not same_for_billing and self.forms[1].fields[field].required and value_field =='':
form.errors[field] = msg_err
non_errors = False
return non_errors
class AddressForm(forms.ModelForm):
class Meta:
model = Address
address1 = forms.CharField()
address2 = forms.CharField(required=False)
zip_code = forms.CharField()
city = forms.CharField()
country = forms.ChoiceField(choices=CountryField.COUNTRIES, initial='FRA')
Ответы
Ответ 1
Каждый набор форм Django поставляется с формой управления, которая должна быть включена в сообщение. Официальные документы объясняют это довольно хорошо. Чтобы использовать его в своем unit test, вам либо нужно написать его самостоятельно. (Ссылка, представленная мной, показывает пример) или вызов formset.management_form
, который выводит данные.
Ответ 2
В частности, я обнаружил, что средство проверки ManagmentForm ищет следующие элементы для отправки:
form_data = {
'form-TOTAL_FORMS': 1,
'form-INITIAL_FORMS': 0
}
Ответ 3
Фактически легко воспроизвести все, что находится в составе, путем проверки контекста ответа.
Рассмотрим приведенный ниже код (с self.client
является регулярным тестовым клиентом):
url = "some_url"
response = self.client.get(url)
self.assertEqual(response.status_code, 200)
# data will receive all the forms field names
# key will be the field name (as "formx-fieldname"), value will be the string representation.
data = {}
# global information, some additional fields may go there
data['csrf_token'] = response.context['csrf_token']
# management form information, needed because of the formset
management_form = response.context['form'].management_form
for i in 'TOTAL_FORMS', 'INITIAL_FORMS', 'MIN_NUM_FORMS', 'MAX_NUM_FORMS':
data['%s-%s' % (management_form.prefix, i)] = management_form[i].value()
for i in range(response.context['form'].total_form_count()):
# get form index 'i'
current_form = response.context['form'].forms[i]
# retrieve all the fields
for field_name in current_form.fields:
value = current_form[field_name].value()
data['%s-%s' % (current_form.prefix, field_name)] = value if value is not None else ''
# flush out to stdout
print '#' * 30
for i in sorted(data.keys()):
print i, '\t:', data[i]
# post the request without any change
response = self.client.post(url, data)
Важное примечание
Если вы изменили data
до вызова self.client.post
, вы, скорее всего, будете мутировать БД. Как следствие, последующий вызов self.client.get
может не привести к тем же данным, в частности для формы управления и порядка форм в наборе форм (поскольку их можно упорядочить по-разному, в зависимости от базового набора запросов). Это означает, что
- если вы изменяете
data[form-3-somefield]
и вызываете self.client.get
, это же поле может появиться в say data[form-8-somefield]
,
- если вы изменили
data
до self.client.post
, вы не можете снова вызвать self.client.post
с тем же data
: вам нужно снова вызвать self.client.get
и восстановить data
.
Ответ 4
Это не похоже на набор форм. Форматы всегда будут иметь какой-то префикс для каждого значения POSTED, а также для ManagementForm, о котором упоминает Бартек. Возможно, это помогло, если вы отправили код просматриваемого вами теста, а также тот вид формы/формы, который он использует.
Ответ 5
Мое дело может быть нарушением, но на некоторых экземплярах на самом деле отсутствует поле, установленное в форме/шаблоне "contrib" администратора, приводящей к ошибке
"Данные ManagementForm отсутствуют или были подделаны"
при сохранении.
Проблема заключалась в методе unicode (SomeModel: [Bad Unicode data]), который я нашел, исследуя строки, которые отсутствовали.
Извлеченный урок заключается в том, чтобы не использовать карту символов MS, я думаю. Моя проблема заключалась в вульгарных фракциях (¼, ½, ¾), но я предполагаю, что это может произойти по-разному. Для специальных символов копирование/вставка с страницы w3 utf-8 исправлена.
postscript-utf-8