Подтвердить что ты не робот

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')
4b9b3361

Ответ 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