Import validation error django

The web framework for perfectionists with deadlines.

Form and field validation¶

Form validation happens when the data is cleaned. If you want to customize
this process, there are various places to make changes, each one serving a
different purpose. Three types of cleaning methods are run during form
processing. These are normally executed when you call the is_valid()
method on a form. There are other things that can also trigger cleaning and
validation (accessing the errors attribute or calling full_clean()
directly), but normally they won’t be needed.

In general, any cleaning method can raise ValidationError if there is a
problem with the data it is processing, passing the relevant information to
the ValidationError constructor. See below
for the best practice in raising ValidationError. If no ValidationError
is raised, the method should return the cleaned (normalized) data as a Python
object.

Most validation can be done using validators — helpers that can be reused.
Validators are functions (or callables) that take a single argument and raise
ValidationError on invalid input. Validators are run after the field’s
to_python and validate methods have been called.

Validation of a form is split into several steps, which can be customized or
overridden:

  • The to_python() method on a Field is the first step in every
    validation. It coerces the value to a correct datatype and raises
    ValidationError if that is not possible. This method accepts the raw
    value from the widget and returns the converted value. For example, a
    FloatField will turn the data into a Python float or raise a
    ValidationError.

  • The validate() method on a Field handles field-specific validation
    that is not suitable for a validator. It takes a value that has been
    coerced to a correct datatype and raises ValidationError on any error.
    This method does not return anything and shouldn’t alter the value. You
    should override it to handle validation logic that you can’t or don’t
    want to put in a validator.

  • The run_validators() method on a Field runs all of the field’s
    validators and aggregates all the errors into a single
    ValidationError. You shouldn’t need to override this method.

  • The clean() method on a Field subclass is responsible for running
    to_python(), validate(), and run_validators() in the correct
    order and propagating their errors. If, at any time, any of the methods
    raise ValidationError, the validation stops and that error is raised.
    This method returns the clean data, which is then inserted into the
    cleaned_data dictionary of the form.

  • The clean_<fieldname>() method is called on a form subclass – where
    <fieldname> is replaced with the name of the form field attribute.
    This method does any cleaning that is specific to that particular
    attribute, unrelated to the type of field that it is. This method is not
    passed any parameters. You will need to look up the value of the field
    in self.cleaned_data and remember that it will be a Python object
    at this point, not the original string submitted in the form (it will be
    in cleaned_data because the general field clean() method, above,
    has already cleaned the data once).

    For example, if you wanted to validate that the contents of a
    CharField called serialnumber was unique,
    clean_serialnumber() would be the right place to do this. You don’t
    need a specific field (it’s a CharField), but you want a
    formfield-specific piece of validation and, possibly, cleaning/normalizing
    the data.

    The return value of this method replaces the existing value in
    cleaned_data, so it must be the field’s value from cleaned_data (even
    if this method didn’t change it) or a new cleaned value.

  • The form subclass’s clean() method can perform validation that requires
    access to multiple form fields. This is where you might put in checks such as
    “if field A is supplied, field B must contain a valid email address”.
    This method can return a completely different dictionary if it wishes, which
    will be used as the cleaned_data.

    Since the field validation methods have been run by the time clean() is
    called, you also have access to the form’s errors attribute which
    contains all the errors raised by cleaning of individual fields.

    Note that any errors raised by your Form.clean() override will not
    be associated with any field in particular. They go into a special
    “field” (called __all__), which you can access via the
    non_field_errors() method if you need to. If you
    want to attach errors to a specific field in the form, you need to call
    add_error().

    Also note that there are special considerations when overriding
    the clean() method of a ModelForm subclass. (see the
    ModelForm documentation for more information)

These methods are run in the order given above, one field at a time. That is,
for each field in the form (in the order they are declared in the form
definition), the Field.clean() method (or its override) is run, then
clean_<fieldname>(). Finally, once those two methods are run for every
field, the Form.clean() method, or its override, is executed whether
or not the previous methods have raised errors.

Examples of each of these methods are provided below.

As mentioned, any of these methods can raise a ValidationError. For any
field, if the Field.clean() method raises a ValidationError, any
field-specific cleaning method is not called. However, the cleaning methods
for all remaining fields are still executed.

Raising ValidationError

In order to make error messages flexible and easy to override, consider the
following guidelines:

  • Provide a descriptive error code to the constructor:

    # Good
    ValidationError(_('Invalid value'), code='invalid')
    
    # Bad
    ValidationError(_('Invalid value'))
    
  • Don’t coerce variables into the message; use placeholders and the params
    argument of the constructor:

    # Good
    ValidationError(
        _('Invalid value: %(value)s'),
        params={'value': '42'},
    )
    
    # Bad
    ValidationError(_('Invalid value: %s') % value)
    
  • Use mapping keys instead of positional formatting. This enables putting
    the variables in any order or omitting them altogether when rewriting the
    message:

    # Good
    ValidationError(
        _('Invalid value: %(value)s'),
        params={'value': '42'},
    )
    
    # Bad
    ValidationError(
        _('Invalid value: %s'),
        params=('42',),
    )
    
  • Wrap the message with gettext to enable translation:

    # Good
    ValidationError(_('Invalid value'))
    
    # Bad
    ValidationError('Invalid value')
    

Putting it all together:

raise ValidationError(
    _('Invalid value: %(value)s'),
    code='invalid',
    params={'value': '42'},
)

Following these guidelines is particularly necessary if you write reusable
forms, form fields, and model fields.

While not recommended, if you are at the end of the validation chain
(i.e. your form clean() method) and you know you will never need
to override your error message you can still opt for the less verbose:

ValidationError(_('Invalid value: %s') % value)

The Form.errors.as_data() and
Form.errors.as_json() methods
greatly benefit from fully featured ValidationErrors (with a code name
and a params dictionary).

Raising multiple errors¶

If you detect multiple errors during a cleaning method and wish to signal all
of them to the form submitter, it is possible to pass a list of errors to the
ValidationError constructor.

As above, it is recommended to pass a list of ValidationError instances
with codes and params but a list of strings will also work:

# Good
raise ValidationError([
    ValidationError(_('Error 1'), code='error1'),
    ValidationError(_('Error 2'), code='error2'),
])

# Bad
raise ValidationError([
    _('Error 1'),
    _('Error 2'),
])

Using validation in practice¶

The previous sections explained how validation works in general for forms.
Since it can sometimes be easier to put things into place by seeing each
feature in use, here are a series of small examples that use each of the
previous features.

Using validators¶

Django’s form (and model) fields support use of utility functions and classes
known as validators. A validator is a callable object or function that takes a
value and returns nothing if the value is valid or raises a
ValidationError if not. These can be passed to a
field’s constructor, via the field’s validators argument, or defined on the
Field class itself with the default_validators
attribute.

Validators can be used to validate values inside the field, let’s have a look
at Django’s SlugField:

from django.core import validators
from django.forms import CharField

class SlugField(CharField):
    default_validators = [validators.validate_slug]

As you can see, SlugField is a CharField with a customized validator
that validates that submitted text obeys to some character rules. This can also
be done on field definition so:

is equivalent to:

slug = forms.CharField(validators=[validators.validate_slug])

Common cases such as validating against an email or a regular expression can be
handled using existing validator classes available in Django. For example,
validators.validate_slug is an instance of
a RegexValidator constructed with the first
argument being the pattern: ^[-a-zA-Z0-9_]+$. See the section on
writing validators to see a list of what is already
available and for an example of how to write a validator.

Form field default cleaning¶

Let’s first create a custom form field that validates its input is a string
containing comma-separated email addresses. The full class looks like this:

from django import forms
from django.core.validators import validate_email

class MultiEmailField(forms.Field):
    def to_python(self, value):
        """Normalize data to a list of strings."""
        # Return an empty list if no input was given.
        if not value:
            return []
        return value.split(',')

    def validate(self, value):
        """Check if value consists only of valid emails."""
        # Use the parent's handling of required fields, etc.
        super().validate(value)
        for email in value:
            validate_email(email)

Every form that uses this field will have these methods run before anything
else can be done with the field’s data. This is cleaning that is specific to
this type of field, regardless of how it is subsequently used.

Let’s create a ContactForm to demonstrate how you’d use this field:

class ContactForm(forms.Form):
    subject = forms.CharField(max_length=100)
    message = forms.CharField()
    sender = forms.EmailField()
    recipients = MultiEmailField()
    cc_myself = forms.BooleanField(required=False)

Use MultiEmailField like any other form field. When the is_valid()
method is called on the form, the MultiEmailField.clean() method will be
run as part of the cleaning process and it will, in turn, call the custom
to_python() and validate() methods.

Cleaning a specific field attribute¶

Continuing on from the previous example, suppose that in our ContactForm,
we want to make sure that the recipients field always contains the address
"fred@example.com". This is validation that is specific to our form, so we
don’t want to put it into the general MultiEmailField class. Instead, we
write a cleaning method that operates on the recipients field, like so:

from django import forms
from django.core.exceptions import ValidationError

class ContactForm(forms.Form):
    # Everything as before.
    ...

    def clean_recipients(self):
        data = self.cleaned_data['recipients']
        if "fred@example.com" not in data:
            raise ValidationError("You have forgotten about Fred!")

        # Always return a value to use as the new cleaned data, even if
        # this method didn't change it.
        return data

Cleaning and validating fields that depend on each other¶

Suppose we add another requirement to our contact form: if the cc_myself
field is True, the subject must contain the word "help". We are
performing validation on more than one field at a time, so the form’s
clean() method is a good spot to do this. Notice that we are
talking about the clean() method on the form here, whereas earlier we were
writing a clean() method on a field. It’s important to keep the field and
form difference clear when working out where to validate things. Fields are
single data points, forms are a collection of fields.

By the time the form’s clean() method is called, all the individual field
clean methods will have been run (the previous two sections), so
self.cleaned_data will be populated with any data that has survived so
far. So you also need to remember to allow for the fact that the fields you
are wanting to validate might not have survived the initial individual field
checks.

There are two ways to report any errors from this step. Probably the most
common method is to display the error at the top of the form. To create such
an error, you can raise a ValidationError from the clean() method. For
example:

from django import forms
from django.core.exceptions import ValidationError

class ContactForm(forms.Form):
    # Everything as before.
    ...

    def clean(self):
        cleaned_data = super().clean()
        cc_myself = cleaned_data.get("cc_myself")
        subject = cleaned_data.get("subject")

        if cc_myself and subject:
            # Only do something if both fields are valid so far.
            if "help" not in subject:
                raise ValidationError(
                    "Did not send for 'help' in the subject despite "
                    "CC'ing yourself."
                )

In this code, if the validation error is raised, the form will display an
error message at the top of the form (normally) describing the problem. Such
errors are non-field errors, which are displayed in the template with
{{ form.non_field_errors }}.

The call to super().clean() in the example code ensures that any validation
logic in parent classes is maintained. If your form inherits another that
doesn’t return a cleaned_data dictionary in its clean() method (doing
so is optional), then don’t assign cleaned_data to the result of the
super() call and use self.cleaned_data instead:

def clean(self):
    super().clean()
    cc_myself = self.cleaned_data.get("cc_myself")
    ...

The second approach for reporting validation errors might involve assigning the
error message to one of the fields. In this case, let’s assign an error message
to both the “subject” and “cc_myself” rows in the form display. Be careful when
doing this in practice, since it can lead to confusing form output. We’re
showing what is possible here and leaving it up to you and your designers to
work out what works effectively in your particular situation. Our new code
(replacing the previous sample) looks like this:

from django import forms

class ContactForm(forms.Form):
    # Everything as before.
    ...

    def clean(self):
        cleaned_data = super().clean()
        cc_myself = cleaned_data.get("cc_myself")
        subject = cleaned_data.get("subject")

        if cc_myself and subject and "help" not in subject:
            msg = "Must put 'help' in subject when cc'ing yourself."
            self.add_error('cc_myself', msg)
            self.add_error('subject', msg)

The second argument of add_error() can be a string, or preferably an
instance of ValidationError. See Raising ValidationError for more
details. Note that add_error() automatically removes the field from
cleaned_data.

Валидация форм и полей¶

Валидация формы происходит при очистке данных. Если вы хотите настроить этот процесс, есть различные места для внесения изменений, каждое из которых служит для разных целей. В процессе обработки формы выполняются три типа методов очистки. Обычно они выполняются, когда вы вызываете метод is_valid() на форме. Есть и другие вещи, которые также могут вызвать очистку и проверку (обращение к атрибуту errors или прямой вызов full_clean()), но обычно они не нужны.

В общем, любой метод очистки может поднять ValidationError, если есть проблема с данными, которые он обрабатывает, передавая соответствующую информацию конструктору ValidationError. See below для лучшей практики поднятия ValidationError. Если не поднимается ValidationError, метод должен вернуть очищенные (нормализованные) данные в виде объекта Python.

Большинство валидаций можно выполнить с помощью validators — помощников, которые можно использовать повторно. Валидаторы — это функции (или callables), которые принимают один аргумент и вызывают ValidationError при недопустимом вводе. Валидаторы запускаются после вызова методов to_python и validate поля.

Валидация формы разбита на несколько этапов, которые можно настроить или отменить:

  • Метод to_python() на Field является первым шагом в каждой валидации. Он преобразует значение к правильному типу данных и выдает сообщение ValidationError, если это невозможно. Этот метод принимает необработанное значение от виджета и возвращает преобразованное значение. Например, FloatField превратит данные в Python float или выдаст ValidationError.

  • Метод validate() на Field обрабатывает специфическую для поля валидацию, которая не подходит для валидатора. Он принимает значение, которое было приведено к правильному типу данных, и при любой ошибке выдает сообщение ValidationError. Этот метод ничего не возвращает и не должен изменять значение. Вы должны переопределить его для обработки логики валидации, которую вы не можете или не хотите поместить в валидатор.

  • Метод run_validators() на поле Field запускает все валидаторы поля и объединяет все ошибки в один ValidationError. Вам не нужно переопределять этот метод.

  • Метод clean() в подклассе Field отвечает за выполнение to_python(), validate() и run_validators() в правильном порядке и распространение их ошибок. Если в любой момент времени какой-либо из методов вызывает ошибку ValidationError, валидация останавливается, и эта ошибка выдается. Этот метод возвращает чистые данные, которые затем вставляются в словарь cleaned_data формы.

  • Метод clean_<fieldname>() вызывается на подклассе формы – где <fieldname> заменяется на имя атрибута поля формы. Этот метод выполняет любую очистку, специфичную для данного атрибута, не связанную с типом поля, которым он является. Этому методу не передаются никакие параметры. Вам нужно будет найти значение поля в self.cleaned_data и помнить, что в этот момент это будет объект Python, а не исходная строка, представленная в форме (она будет в cleaned_data, потому что метод general field clean(), описанный выше, уже однажды очистил данные).

    Например, если вы хотите проверить, что содержимое CharField под названием serialnumber является уникальным, clean_serialnumber() будет подходящим местом для этого. Вам не нужно конкретное поле (это CharField), но вам нужен специфический для поля формы фрагмент проверки и, возможно, очистки/нормализации данных.

    Возвращаемое значение этого метода заменяет существующее значение в cleaned_data, поэтому это должно быть значение поля из cleaned_data (даже если этот метод не изменил его) или новое очищенное значение.

  • Метод clean() подкласса формы может выполнять валидацию, требующую доступа к нескольким полям формы. Сюда можно отнести такие проверки, как «если поле A предоставлено, то поле B должно содержать действительный адрес электронной почты». При желании этот метод может вернуть совершенно другой словарь, который будет использован в качестве cleaned_data.

    Поскольку методы валидации полей были запущены к моменту вызова clean(), у вас также есть доступ к атрибуту errors формы, который содержит все ошибки, возникшие при очистке отдельных полей.

    Обратите внимание, что любые ошибки, возникающие при переопределении Form.clean(), не будут связаны с каким-либо конкретным полем. Они попадают в специальное «поле» (называемое __all__), к которому вы можете получить доступ через метод non_field_errors(), если вам это необходимо. Если вы хотите прикрепить ошибки к определенному полю формы, вам нужно вызвать add_error().

    Также обратите внимание, что существуют особые соображения при переопределении метода clean() подкласса ModelForm. (см. ModelForm documentation для получения дополнительной информации)

Эти методы выполняются в указанном выше порядке, по одному полю за раз. То есть, для каждого поля формы (в порядке их объявления в определении формы) выполняется метод Field.clean() (или его переопределение), затем clean_<fieldname>(). Наконец, когда эти два метода выполнены для каждого поля, выполняется метод Form.clean(), или его переопределение, независимо от того, вызвали ли предыдущие методы ошибки.

Примеры каждого из этих методов приведены ниже.

Как уже упоминалось, любой из этих методов может вызвать ошибку ValidationError. Для любого поля, если метод Field.clean() вызывает ValidationError, любой метод очистки, специфичный для данного поля, не вызывается. Однако методы очистки для всех оставшихся полей все равно выполняются.

Поднятие ValidationError

Чтобы сделать сообщения об ошибках гибкими и легко переопределяемыми, примите во внимание следующие рекомендации:

  • Предоставить описательную ошибку code конструктору:

    # Good
    ValidationError(_('Invalid value'), code='invalid')
    
    # Bad
    ValidationError(_('Invalid value'))
    
  • Не вставляйте переменные в сообщение; используйте заполнители и аргумент params конструктора:

    # Good
    ValidationError(
        _('Invalid value: %(value)s'),
        params={'value': '42'},
    )
    
    # Bad
    ValidationError(_('Invalid value: %s') % value)
    
  • Используйте ключи отображения вместо позиционного форматирования. Это позволяет располагать переменные в любом порядке или вообще их не использовать при переписывании сообщения:

    # Good
    ValidationError(
        _('Invalid value: %(value)s'),
        params={'value': '42'},
    )
    
    # Bad
    ValidationError(
        _('Invalid value: %s'),
        params=('42',),
    )
    
  • Оберните сообщение символом gettext, чтобы включить перевод:

    # Good
    ValidationError(_('Invalid value'))
    
    # Bad
    ValidationError('Invalid value')
    

Собираем все вместе:

raise ValidationError(
    _('Invalid value: %(value)s'),
    code='invalid',
    params={'value': '42'},
)

Следование этим рекомендациям особенно необходимо, если вы пишете многократно используемые формы, поля форм и поля моделей.

Хотя это и не рекомендуется, если вы находитесь в конце цепочки валидации (т.е. ваша форма clean() метод) и вы знаете, что вам никогда не понадобится переопределять сообщение об ошибке, вы можете выбрать менее многословный вариант:

ValidationError(_('Invalid value: %s') % value)

Методы Form.errors.as_data() и Form.errors.as_json() значительно выигрывают от полнофункциональных ValidationErrors (с code именем и params словарем).

Возникновение множества ошибок¶

Если вы обнаружили несколько ошибок во время работы метода очистки и хотите сигнализировать обо всех из них отправителю формы, можно передать список ошибок конструктору ValidationError.

Как и выше, рекомендуется передавать список экземпляров ValidationError с codes и params, но подойдет и список строк:

# Good
raise ValidationError([
    ValidationError(_('Error 1'), code='error1'),
    ValidationError(_('Error 2'), code='error2'),
])

# Bad
raise ValidationError([
    _('Error 1'),
    _('Error 2'),
])

Использование валидации на практике¶

В предыдущих разделах объяснялось, как работает валидация в целом для форм. Поскольку иногда бывает проще понять, как работает каждая функция, здесь приведена серия небольших примеров, в которых используется каждая из предыдущих функций.

Использование валидаторов¶

Поля формы (и модели) Django поддерживают использование полезных функций и классов, известных как валидаторы. Валидатор — это вызываемый объект или функция, которая принимает значение и не возвращает ничего, если значение действительно, или выдает ошибку ValidationError, если нет. Они могут быть переданы в конструктор поля через аргумент validators или определены в самом классе Field с помощью атрибута default_validators.

Валидаторы могут использоваться для проверки значений внутри поля, давайте посмотрим на Django’s SlugField:

from django.core import validators
from django.forms import CharField

class SlugField(CharField):
    default_validators = [validators.validate_slug]

Как вы можете видеть, SlugField — это CharField с настроенным валидатором, который проверяет, что отправленный текст соответствует некоторым правилам символов. Это также можно сделать при определении поля так:

эквивалентно:

slug = forms.CharField(validators=[validators.validate_slug])

Обычные случаи, такие как проверка по электронной почте или регулярному выражению, могут быть обработаны с помощью существующих классов валидаторов, доступных в Django. Например, validators.validate_slug — это экземпляр RegexValidator, построенный с первым аргументом в виде шаблона: ^[-a-zA-Z0-9_]+$. Смотрите раздел writing validators, чтобы увидеть список того, что уже доступно, и пример того, как написать валидатор.

Очистка полей формы по умолчанию¶

Давайте сначала создадим поле пользовательской формы, которое проверяет, что его входные данные — это строка, содержащая адреса электронной почты, разделенные запятыми. Полный класс выглядит следующим образом:

from django import forms
from django.core.validators import validate_email

class MultiEmailField(forms.Field):
    def to_python(self, value):
        """Normalize data to a list of strings."""
        # Return an empty list if no input was given.
        if not value:
            return []
        return value.split(',')

    def validate(self, value):
        """Check if value consists only of valid emails."""
        # Use the parent's handling of required fields, etc.
        super().validate(value)
        for email in value:
            validate_email(email)

В каждой форме, использующей это поле, эти методы будут выполняться до того, как с данными поля можно будет сделать что-либо еще. Это очистка, специфичная для данного типа поля, независимо от того, как оно будет использоваться в дальнейшем.

Давайте создадим ContactForm, чтобы продемонстрировать, как вы будете использовать это поле:

class ContactForm(forms.Form):
    subject = forms.CharField(max_length=100)
    message = forms.CharField()
    sender = forms.EmailField()
    recipients = MultiEmailField()
    cc_myself = forms.BooleanField(required=False)

Используйте MultiEmailField как любое другое поле формы. Когда на форме будет вызван метод is_valid(), в процессе очистки будет запущен метод MultiEmailField.clean(), который, в свою очередь, вызовет пользовательские методы to_python() и validate().

Очистка определенного атрибута поля¶

Продолжая предыдущий пример, предположим, что в нашем ContactForm мы хотим убедиться, что поле recipients всегда содержит адрес "fred@example.com". Это проверка, специфичная для нашей формы, поэтому мы не хотим помещать ее в общий класс MultiEmailField. Вместо этого мы напишем метод очистки, который работает с полем recipients, следующим образом:

from django import forms
from django.core.exceptions import ValidationError

class ContactForm(forms.Form):
    # Everything as before.
    ...

    def clean_recipients(self):
        data = self.cleaned_data['recipients']
        if "fred@example.com" not in data:
            raise ValidationError("You have forgotten about Fred!")

        # Always return a value to use as the new cleaned data, even if
        # this method didn't change it.
        return data

Очистка и проверка полей, которые зависят друг от друга¶

Предположим, мы добавим еще одно требование к нашей контактной форме: если поле cc_myself является True, то subject должно содержать слово "help". Мы выполняем проверку более чем одного поля одновременно, поэтому метод формы clean() является хорошим местом для этого. Обратите внимание, что здесь мы говорим о методе clean() на форме, тогда как ранее мы писали метод clean() на поле. Важно четко различать поля и формы, когда мы решаем, где проводить валидацию. Поля — это отдельные точки данных, а формы — это набор полей.

К моменту вызова метода clean() формы будут запущены все методы очистки отдельных полей (предыдущие два раздела), поэтому self.cleaned_data будет заполнен любыми данными, которые сохранились до сих пор. Поэтому вам также нужно помнить о том, что поля, которые вы хотите проверить, могут не выдержать первоначальной проверки отдельных полей.

Есть два способа сообщить о любых ошибках на этом этапе. Вероятно, самый распространенный способ — вывести ошибку в верхней части формы. Чтобы создать такую ошибку, вы можете поднять ValidationError из метода clean(). Например:

from django import forms
from django.core.exceptions import ValidationError

class ContactForm(forms.Form):
    # Everything as before.
    ...

    def clean(self):
        cleaned_data = super().clean()
        cc_myself = cleaned_data.get("cc_myself")
        subject = cleaned_data.get("subject")

        if cc_myself and subject:
            # Only do something if both fields are valid so far.
            if "help" not in subject:
                raise ValidationError(
                    "Did not send for 'help' in the subject despite "
                    "CC'ing yourself."
                )

В этом коде, если возникает ошибка валидации, форма выводит сообщение об ошибке в верхней части формы (обычно) с описанием проблемы. Такие ошибки являются не-полевыми ошибками, которые отображаются в шаблоне с помощью {{ form.non_field_errors }}.

Вызов super().clean() в коде примера гарантирует, что любая логика валидации в родительских классах будет сохранена. Если ваша форма наследует другую, которая не возвращает словарь cleaned_data в своем методе clean() (это необязательно), то не присваивайте cleaned_data результату вызова super() и используйте self.cleaned_data вместо этого:

def clean(self):
    super().clean()
    cc_myself = self.cleaned_data.get("cc_myself")
    ...

Второй подход для сообщения об ошибках валидации может включать присвоение сообщения об ошибке одному из полей. В данном случае давайте присвоим сообщение об ошибке обеим строкам «subject» и «cc_myself» в отображении формы. Будьте осторожны, делая это на практике, так как это может привести к запутанному выводу формы. Мы показываем, что здесь возможно, и предоставляем вам и вашим дизайнерам самим решать, что будет эффективно работать в вашей конкретной ситуации. Наш новый код (заменяющий предыдущий пример) выглядит следующим образом:

from django import forms

class ContactForm(forms.Form):
    # Everything as before.
    ...

    def clean(self):
        cleaned_data = super().clean()
        cc_myself = cleaned_data.get("cc_myself")
        subject = cleaned_data.get("subject")

        if cc_myself and subject and "help" not in subject:
            msg = "Must put 'help' in subject when cc'ing yourself."
            self.add_error('cc_myself', msg)
            self.add_error('subject', msg)

Вторым аргументом add_error() может быть строка или, предпочтительно, экземпляр ValidationError. Более подробную информацию смотрите в Поднятие ValidationError. Обратите внимание, что add_error() автоматически удаляет поле из cleaned_data.

Проверка формы и поля

Проверка формы происходит при очистке данных. Если вы хотите настроить этот процесс, есть несколько мест для внесения изменений, каждое из которых служит своей цели. Во время обработки формы запускаются три типа методов очистки. Обычно они выполняются, когда вы вызываете метод is_valid() в форме. Есть и другие вещи, которые также могут вызвать очистку и проверку (доступ к атрибуту errors или прямой вызов full_clean() ), но обычно они не нужны.

В общем, любой метод очистки может вызвать ValidationError , если есть проблема с данными, которые он обрабатывает, передав соответствующую информацию конструктору ValidationError . Ниже приведены рекомендации по повышению ValidationError . Если ValidationError не вызывается, метод должен вернуть очищенные (нормализованные) данные в виде объекта Python.

Большую часть проверки можно выполнить с помощью валидаторов — помощников, которые можно использовать повторно. Валидаторы — это функции (или вызываемые объекты), которые принимают один аргумент и вызывают ValidationError при неверном вводе. Валидаторы запускаются после вызова методов поля to_python и validate .

Проверка формы разделена на несколько этапов,которые можно настроить или переопределить:

  • Метод to_python() для Field является первым шагом в каждой проверке. Он приводит значение к правильному типу данных и вызывает ValidationError , если это невозможно. Этот метод принимает необработанное значение из виджета и возвращает преобразованное значение. Например, FloatField превратит данные в число с float Python или вызовет ValidationError .
  • Метод validate() для Field обрабатывает специфическую для поля проверку, которая не подходит для валидатора. Он принимает значение, которое было приведено к правильному типу данных, и вызывает ValidationError при любой ошибке. Этот метод ничего не возвращает и не должен изменять значение. Вы должны переопределить его, чтобы обрабатывать логику проверки, которую вы не можете или не хотите использовать в валидаторе.
  • Метод run_validators() для Field запускает все валидаторы поля и объединяет все ошибки в один ValidationError . Вам не нужно переопределять этот метод.
  • Метод clean() в подклассе Field отвечает за запуск to_python() , validate() и run_validators() в правильном порядке и распространение их ошибок. Если в любое время любой из методов вызывает ValidationError , проверка останавливается, и возникает эта ошибка. Этот метод возвращает чистые данные, которые затем вставляются в словарь cleaned_data формы.
  • Метод clean_<fieldname>() вызывается в подклассе формы, где <fieldname> заменяется именем атрибута поля формы. Этот метод выполняет любую очистку, относящуюся к этому конкретному атрибуту, независимо от типа поля, которым он является. Этому методу не передаются никакие параметры. Вам нужно будет найти значение поля в self.cleaned_data и помнить, что в этот момент это будет объект Python, а не исходная строка, представленная в форме (она будет в cleaned_data потому что метод clean() общего поля , выше, уже один раз очистил данные).

    Например, если вы хотите проверить уникальность содержимого CharField , называемого serialnumber clean_serialnumber() . Вам не нужно конкретное поле (это CharField ), но вам нужна часть проверки, специфичная для поля формы, и, возможно, очистка/нормализация данных.

    Возвращаемое значение этого метода заменяет существующее значение в cleaned_data , поэтому оно должно быть значением поля из cleaned_data (даже если этот метод не изменил его) или новым очищенным значением.

  • clean() подкласса формы может выполнять проверку, требующую доступа к нескольким полям формы. Здесь вы можете поставить такие проверки, как «если поле A указано, поле B должно содержать действительный адрес электронной почты». Этот метод может вернуть совершенно другой словарь, если захочет, который будет использоваться как cleaned_data .

    Поскольку методы проверки поля были запущены к моменту clean() , у вас также есть доступ к атрибуту errors формы, который содержит все ошибки, вызванные очисткой отдельных полей.

    Обратите внимание, что любые ошибки, вызванные вашим переопределением Form.clean() , не будут связаны с каким-либо конкретным полем. Они попадают в специальное «поле» (называемое __all__ ), к которому вы можете получить доступ через метод non_field_errors() , если вам это нужно. Если вы хотите прикрепить ошибки к определенному полю в форме, вам нужно вызвать add_error() .

    Также обратите внимание, что при переопределении метода clean() подкласса ModelForm есть особые соображения . (см. документацию ModelForm для получения дополнительной информации)

Эти методы выполняются в порядке, указанном выше, по одному полю за раз. То есть для каждого поля в форме (в порядке их объявления в определении формы) Field.clean() метод Field.clean () (или его переопределение), затем clean_<fieldname>() . Наконец, после того, как эти два метода запущены для каждого поля, Form.clean() метод Form.clean () или его переопределение независимо от того, были ли в предыдущих методах ошибки.

Примеры каждого из этих методов приведены ниже.

Как уже упоминалось, любой из этих методов может вызвать ValidationError . Для любого поля, если метод Field.clean() вызывает ValidationError , какой-либо специфический для поля метод очистки не вызывается. Однако методы очистки для всех оставшихся полей все еще выполняются.

Raising ValidationError

Для того,чтобы сделать сообщения об ошибках гибкими и легко преодолеваемыми,обратите внимание на следующие рекомендации:

  • Предоставьте описательный code ошибки конструктору:

    ValidationError(_('Invalid value'), code='invalid')
    
    
    ValidationError(_('Invalid value'))
    
  • Не добавляйте переменные в сообщение принудительно; используйте заполнители и аргумент params конструктора:

    # Good
    ValidationError(
        _('Invalid value: %(value)s'),
        params={'value': '42'},
    )
    
    # Bad
    ValidationError(_('Invalid value: %s') % value)
    
  • Используйте клавиши отображения вместо позиционного форматирования.Это позволяет расположить переменные в любом порядке или вообще опустить их при переписывании сообщения:

    # Good
    ValidationError(
        _('Invalid value: %(value)s'),
        params={'value': '42'},
    )
    
    # Bad
    ValidationError(
        _('Invalid value: %s'),
        params=('42',),
    )
    
  • Оберните сообщение gettext , чтобы включить перевод:

    ValidationError(_('Invalid value'))
    
    
    ValidationError('Invalid value')
    

Складывая все вместе:

raise ValidationError(
    _('Invalid value: %(value)s'),
    code='invalid',
    params={'value': '42'},
)

Соблюдение этих рекомендаций особенно необходимо,если Вы пишете формы многоразового использования,поля форм и поля моделей.

Хотя это и не рекомендуется, если вы находитесь в конце цепочки валидации (т. Е. Метода формы clean() ) и знаете, что вам никогда не понадобится переопределять сообщение об ошибке, вы все равно можете выбрать менее подробный:

ValidationError(_('Invalid value: %s') % value)

Form.errors.as_data() и Form.errors.as_json() метода извлечь большую пользу из полнофункционального ValidationError s (с code именем и params словарь).

Повышение количества ошибок

Если вы обнаружите несколько ошибок во время метода очистки и хотите сообщить обо всех из них отправителю формы, можно передать список ошибок конструктору ValidationError .

Как и выше, рекомендуется передавать список экземпляров ValidationError с code s и params , но также будет работать список строк:

# Good
raise ValidationError([
    ValidationError(_('Error 1'), code='error1'),
    ValidationError(_('Error 2'), code='error2'),
])

# Bad
raise ValidationError([
    _('Error 1'),
    _('Error 2'),
])

Использование валидации на практике

В предыдущих разделах объяснялось,как работает валидация в целом для форм.Так как иногда может быть проще поставить вещи на место,видя каждую используемую функцию,вот серия маленьких примеров,которые используют каждую из предыдущих функций.

Using validators

Поля формы (и модели) Django поддерживают использование служебных функций и классов, известных как валидаторы. Валидатор — это вызываемый объект или функция, которая принимает значение и ничего не возвращает, если значение допустимо, или вызывает ValidationError , если нет. Их можно передать конструктору поля через аргумент validators поля или определить в самом классе Field с помощью атрибута default_validators .

Валидаторы можно использовать для проверки значений внутри поля, давайте взглянем на SlugField Django :

from django.core import validators
from django.forms import CharField

class SlugField(CharField):
    default_validators = [validators.validate_slug]

Как видите, SlugField — это CharField с настроенным валидатором, который проверяет, что отправленный текст подчиняется некоторым правилам символов. Это также можно сделать для определения поля так:

slug = forms.SlugField()

эквивалентно:

slug = forms.CharField(validators=[validators.validate_slug])

Распространенные случаи, такие как проверка по электронной почте или регулярному выражению, могут быть обработаны с использованием существующих классов проверки, доступных в Django. Например, validators.validate_slug является экземпляром RegexValidator , построенным с первым аргументом, являющимся шаблоном: ^[-a-zA-Z0-9_]+$ . См. Раздел о написании валидаторов, чтобы увидеть список того, что уже доступно, и пример того, как написать валидатор.

Очистка полей формы по умолчанию

Давайте сначала создадим поле пользовательской формы,которое проверяет,что его входные данные-это строка,содержащая адреса электронной почты,разделенные запятыми.Полный класс выглядит следующим образом:

from django import forms
from django.core.validators import validate_email

class MultiEmailField(forms.Field):
    def to_python(self, value):
        """Normalize data to a list of strings."""
        
        if not value:
            return []
        return value.split(',')

    def validate(self, value):
        """Check if value consists only of valid emails."""
        
        super().validate(value)
        for email in value:
            validate_email(email)

В каждой форме,использующей это поле,эти методы будут выполняться до того,как с данными поля можно будет сделать что-либо еще.Это очистка,специфичная для данного типа поля,независимо от того,как оно будет использоваться в дальнейшем.

Давайте создадим ContactForm , чтобы продемонстрировать, как вы будете использовать это поле:

class ContactForm(forms.Form):
    subject = forms.CharField(max_length=100)
    message = forms.CharField()
    sender = forms.EmailField()
    recipients = MultiEmailField()
    cc_myself = forms.BooleanField(required=False)

Используйте MultiEmailField , как и любое другое поле формы. Когда is_valid() метод вызывается на форме, MultiEmailField.clean() метод будет работать как часть процесса очистки , и это, в свою очередь, вызывать пользовательские to_python() и validate() методы.

Очистка конкретного атрибута поля

Продолжая предыдущий пример, предположим, что в нашей ContactForm мы хотим убедиться, что поле recipients всегда содержит адрес "fred@example.com" . Это проверка, специфичная для нашей формы, поэтому мы не хотим помещать ее в общий класс MultiEmailField . Вместо этого мы пишем метод очистки, который работает с полем recipients

from django import forms
from django.core.exceptions import ValidationError

class ContactForm(forms.Form):
    
    ...

    def clean_recipients(self):
        data = self.cleaned_data['recipients']
        if "fred@example.com" not in data:
            raise ValidationError("You have forgotten about Fred!")

        
        
        return data

Очистка и проверка полей,которые зависят друг от друга.

Предположим, мы добавляем еще одно требование к нашей контактной форме: если поле cc_myself равно True , subject должна содержать слово "help" . Мы выполняем проверку более чем одного поля одновременно, поэтому метод clean() формы является хорошим местом для этого. Обратите внимание, что здесь мы говорим о методе clean() для формы, тогда как ранее мы писали метод clean() для поля. Важно сохранять четкость поля и различия формы при работе над тем, где проверять вещи. Поля — это отдельные точки данных, формы — это набор полей.

К тому времени, когда будет вызван метод clean() формы , все методы очистки отдельных полей будут запущены (предыдущие два раздела), поэтому self.cleaned_data будет заполнен всеми данными, которые сохранились до сих пор. Поэтому вам также необходимо помнить о том, что поля, которые вы хотите проверить, могли не пройти первоначальную проверку отдельных полей.

На этом этапе есть два способа сообщить о любых ошибках. Вероятно, наиболее распространенный способ — отобразить ошибку в верхней части формы. Чтобы создать такую ​​ошибку, вы можете вызвать ValidationError из метода clean() . Например:

from django import forms
from django.core.exceptions import ValidationError

class ContactForm(forms.Form):
    
    ...

    def clean(self):
        cleaned_data = super().clean()
        cc_myself = cleaned_data.get("cc_myself")
        subject = cleaned_data.get("subject")

        if cc_myself and subject:
            
            if "help" not in subject:
                raise ValidationError(
                    "Did not send for 'help' in the subject despite "
                    "CC'ing yourself."
                )

В этом коде, если возникает ошибка проверки, форма будет отображать сообщение об ошибке вверху формы (обычно) с описанием проблемы. Такие ошибки не относятся к полевым ошибкам, которые отображаются в шаблоне с помощью {{ form.non_field_errors }} .

Вызов super().clean() в коде примера гарантирует, что любая логика проверки в родительских классах сохраняется. Если ваша форма наследует другую, которая не возвращает словарь cleaned_data в своем методе clean() (это необязательно), то не назначайте cleaned_data результату вызова super() и вместо этого используйте self.cleaned_data :

def clean(self):
    super().clean()
    cc_myself = self.cleaned_data.get("cc_myself")
    ...

Второй подход для сообщения об ошибках валидации может включать присвоение сообщения об ошибке одному из полей.В данном случае давайте присвоим сообщение об ошибке обеим строкам «subject» и «cc_myself» в отображении формы.Будьте осторожны,делая это на практике,так как это может привести к запутанному выводу формы.Мы показываем,что здесь возможно,и предоставляем вам и вашим дизайнерам самим решать,что будет эффективно работать в вашей конкретной ситуации.Наш новый код (заменяющий предыдущий пример)выглядит следующим образом:

from django import forms

class ContactForm(forms.Form):
    
    ...

    def clean(self):
        cleaned_data = super().clean()
        cc_myself = cleaned_data.get("cc_myself")
        subject = cleaned_data.get("subject")

        if cc_myself and subject and "help" not in subject:
            msg = "Must put 'help' in subject when cc'ing yourself."
            self.add_error('cc_myself', msg)
            self.add_error('subject', msg)

Второй аргумент add_error() может быть строкой или предпочтительно экземпляром ValidationError . См. Повышение ValidationError для получения дополнительной информации. Обратите внимание, что add_error() автоматически удаляет поле из cleaned_data .


Django

4.1

  • Форма модели Функции

    Ссылка на API API модели формы.

  • API рендеринга формы

    Виджеты форм в Django отображаются с помощью системы шаблонизаторов.

  • Widgets

    Виджет-это представление Django элемента ввода HTML.

  • Built-in widgets

    Django предоставляет представление всех основных HTML виджетов,плюс некоторые часто используемые группы в модуле django.forms.widgets,включая текст ввода,различные

In this article, we will learn about the Built-In Django Form Validation offered as well as the Django Custom Form Validations.

Prerequisites for Django Form Validation

Before moving forward with Form Validation, you will need to know what Django forms and how you implement them in Django.

If you are already familiar with Django Forms, then continue with the article or else do check out Django Forms article first.

Built-In Form Validations

Django framework provides a list of built-in form validators along with the Forms library. We will learn about each one of them here.

Setting up a Form and Using Built-in Validators

Field = forms.Field_Type(validator = value)

All of the built-in validators follow the above syntax.

The Form View used in this section

The following explanation codes are based on the following View in views.py

def EFormView(request):
    if request.method == 'POST':
        form = EForm(request.POST)
        if form.is_valid():
            form.save()
            return HttpResponse('Form saved')
    else:
        form = EForm()
        context = {
            'form':form,
        }
    return render(request, 'books_website/Form.html', context)

And the Template is a simple HTML file including the form attribute of HTML

<form method ='post'>
    {% csrf_token %}
    {{form.as_p}}
    <input type="submit" value = "Submit">
</form>

1. initial – Placeholder Text with the “Initial” Attribute

You might have seen many forms on the internet come with a pre-entered text like the one given below.

Example
Example

This is done using the initial attribute of Django Forms.

Therefore, if you need a pre-entered text in your form fields, we use the initial attribute.

The initial default value is empty.

class EForm(forms.Form):
    Name = forms.CharField(initial = "Enter Name")
Initial
Initial

2. label – Adding Labels to a Form Field

The label attribute gives a custom name to the form field. As a default, Django uses the form field name specified in forms.py itself as the field name. But using label we can change it to any other custom name.

class EForm(forms.Form):
    Name = forms.CharField(label = "Full Name")
Label
Label

3. max_length – Limit Maximum Length of Characters Entered

The max_length attribute ensures that the information entered in the field does not exceed more than the specified value.

class EForm(forms.Form):
    username = forms.CharField(max_length = 10)
Max Length
Max Length

Note that you wont be able to add more than 10 characters.

4. error_message – Add a Custom Error Message

The error_message attribute lets you add custom error messages to the fields. This attribute overrides the default error message and enables you to write your own.

The error message is passed as a dictionary, as shown:

class EForm(forms.Form):
    username = forms.CharField(error_message = {'required':'Please Enter your Name',})
Image 26
Error Message

5. disabled – Add Uneditable Values to Django Forms

In many forms, specific fields are uneditable by users like the company’s name on specific company forms.

The disabled attribute lets you do that. Using this, you can disable a particular field, thereby leaving it uneditable by the user.

class ITJobForm(forms.Form):
    Interview domain = forms.CharField(disabled = True, initial = 'Software Engineer')
    Name = forms.CharField(max_length = 80)
Disabled
Disabled

Note that you wont be able to change the value for the Interview domain field.

6. help_text – Suggest What Needs to be Entered in the Input

This attribute adds a help text beside the field to give more information about what has to be entered.

class EForm(forms.Form):
    Name = forms.CharField(help_text ='Please enter your name')
Help Text
Help Text

7. required – Ensure a Form Field has Data Before Submitting

In many forms, certain fields are compulsory, like the applicant’s name, number, etc. This is done with the required attribute.

Default value is False

class EForm(forms.Form):
    Name = forms.CharField(required = True)
Fullscreen 1
Required

8. widget – Set up Individual Form Fields for Predefined Purposes

Widget is Django’s representation of HTML input. Each field has its predefined field widget. For example, a Number Field will have a widget taking only number input, email field taking only the email input, etc.

We can override the default widgets and add our own as shown:

class EForm(forms.Form):
    Name = forms.CharField(widget = forms.Textarea)
Widget
Widget

Custom Form Validation using ValidationError attribute

As seen in the Django exception handling article, the Validation Error attribute is used to validate custom form information entered by the user.

The syntax for using Validation Error

ValidationError('text',params = {value : <value>})

Here, the value variable and the text depends on where we are calling the attribute and also on the condition for validation.

The full syntax for ValidationError is:

import django.core.exceptions
if cond:
    raise ValidationError('text')

Implementing of Custom Validations

Lets us create a form that allows username to be between 10 to 20 characters only. The form.py code will be:

class EForm(forms.Form):
    Name = forms.CharField(label = "Full Name")

Add the Form View Code as shown:

from django.core.exceptions import ValidationError
def FormView(request):
    if request.method == 'POST':
        form = EForm(request.POST)
        if form.is_valid():
            Name = form.cleaned_data['Name']
            if len(Name)< 10 or len(Name) > 20:
                raise ValidationError(f'Length of the name:{Name} is not between 10 -20 characters')
            return HttpResponse('Form saved')
    else:
        form = EForm()
        context ={
            'form':form
        }
    return render(request,'books_website/Form.html',context)

Notice how the Validation Error is raised. The template is the same as the one seen above.

Let’s run the server and enter a name with less than 10 characters. You can also try the same with more than 10 characters as it still voids our condition.

Validation Error 1
Validation Error

When you hit submit, you’ll get an error as shown below:

Error
Form Error

Now we will try for a name in between 10 to 20 characters

Form 10
Form 11 characters

Hit submit and check

Saved
Form Saved

See how the form accepts only certain names. This is how a ValidationError works.

Conclusion

That’s it, guys !! This was all about form validation. Also, check out the official Django forms documentation for more information. Keep practicing!!

Django comes with a useful set of utilities for testing
various aspects of web apps. For example, its TestCase and TransactionTestCase
base classes as well as its test client make it easier to test those cases.

However the Django testing utilities can’t cover everything, so there are many cases
where a robust test involves a lot more manual work than ideal. One of these cases that
crops up pretty often is in testing form and model validation, and in particular,
asserting exactly which validation errors happen.

Suppose we have the following model:

from django import models

class Person(models.Model):
    name = models.CharField(max_length=100)
    dob = models.DateField()
    website = models.URLField(required=False)

    def clean(self):
        if self.name == 'Joe' and self.website is None:
            raise ValidationError({
                'website': "Joe must have a website"
            })

If we want to test that our custom validation is done correctly, we might try
to use assertRaises and do the following:

from django.core.validation import ValidationError
from django.test import TestCase

class PersonTest(TestCase):

    def test_joe_must_have_a_website(self):
        p = Person(name='Joe')
        with self.assertRaises(ValidationError):
            p.full_clean()

This test contains a bug. Can you spot it? The problem is that our model has additional
required fields (dob in our case) which we haven’t specified in the test, so it will
always raise a validation error, even if our code is not correct.

If this was a Form test, we might use assertFormError. For model tests, there’s no
such shortcut so we’re forced to go about it manually:

from django.core.validation import ValidationError
from django.test import TestCase

class PersonTest(TestCase):

    def test_joe_must_have_a_website(self):
        p = Person(name='Joe', dob=datetime.date(1990, 1, 1))

        try:
            p.full_clean()
        except ValidationError as e:
            self.assertTrue('name' in e.message_dict)

If you have a lot of model-level validation, it can be cumbersome to manually use this
pattern all the time. Instead, you can extract the pattern into a helper:

class ValidationErrorTestMixin(object):

    @contextmanager
    def assertValidationErrors(self, fields):
        """
        Assert that a validation error is raised, containing all the specified
        fields, and only the specified fields.
        """
        try:
            yield
            raise AssertionError("ValidationError not raised")
        except ValidationError as e:
            self.assertEqual(set(fields), set(e.message_dict.keys()))

You can then use it as:

class PersonTest(ValidationErrorTestMixin, TestCase):

    def test_joe_must_have_a_website(self):
        p = Person(name='Joe', dob=datetime.date(1990, 1, 1))

        with self.assertValidationErrors(self, ['name']):
            p.full_clean()

In cases where you raise validation errors that are not specific to a field, you
can check for NON_FIELD_ERRORS:

from django.core.exceptions import NON_FIELD_ERRORS

class Person(models.Model):
    name = models.CharField(max_length=100)
    dob = models.DateField()
    website = models.URLField(required=False)

    def clean(self):
        if self.name == 'Joe' and self.website is None:
            raise ValidationError("Joe must have a website")


class PersonTest(ValidationErrorTestMixin, TestCase):

    def test_joe_must_have_a_website(self):
        p = Person(name='Joe', dob=datetime.date(1990, 1, 1))

        with self.assertValidationErrors(self, [NON_FIELD_ERRORS]):
            p.full_clean()

If you do a lot of model-level validation, extracting the validation pattern
into assertValidationError as shown here will make your tests more robust,
easier to write, more readable and maintainable.

Author

Senko Rašić

We’re small, experienced and passionate team of web developers, doing custom app development and web consulting.

Prerequisites: Django Installation | Introduction to Django
Django works on an MVT pattern. So there is a need to create data models (or tables). For every table, a model class is created. 
Suppose there is a form that takes Username, gender, and text as input from the user, the task is to validate the data and save it.
In django this can be done, as follows:
 

Python

from django.db import models

class Post(models.Model):

    Male = 'M'

    FeMale = 'F'

    GENDER_CHOICES = (

    (Male, 'Male'),

    (FeMale, 'Female'),

    )

    username = models.CharField( max_length = 20, blank = False,

                                 null = False)

    text = models.TextField(blank = False, null = False)

    gender = models.CharField(max_length = 6, choices = GENDER_CHOICES,

                              default = Male)

    time = models.DateTimeField(auto_now_add = True)

After creating the data models, the changes need to be reflected in the database to do this run the following command:
 

python manage.py makemigrations

Doing this compiles the models and if it didn’t find any errors then, it creates a file in the migration folder. Later run the command given below to finally reflect the changes saved onto the migration file onto the database.
 

python manage.py migrate

Now a form can be created. Suppose that the username length should not be less than 5 and post length should be greater than 10. Then we define the Class PostForm with the required validation rules as follows: 
 

Python

from django.forms import ModelForm

from django import forms

from formValidationApp.models import *

class PostForm(ModelForm):

    class Meta:

        model = Post       

        fields =["username", "gender", "text"]

    def clean(self):

        super(PostForm, self).clean()

        username = self.cleaned_data.get('username')

        text = self.cleaned_data.get('text')

        if len(username) < 5:

            self._errors['username'] = self.error_class([

                'Minimum 5 characters required'])

        if len(text) <10:

            self._errors['text'] = self.error_class([

                'Post Should Contain a minimum of 10 characters'])

        return self.cleaned_data

Till now, the data models and the Form class are defined. Now the focus will be on how these modules, defined above, are actually used.
First, run on localhost through this command 
 

python manage.py runserver

Open http://localhost:8000/ in the browser, then it’s going to search in the urls.py file, looking for ‘ ‘ path
urls.py file is as given below: 
 

Python

from django.contrib import admin

from django.urls import path, include

from django.conf.urls import url

from django.shortcuts import HttpResponse

from . import views

urlpatterns = [

    path('', views.home, name ='index'),

]

Basically, this associates the ‘ ‘ url with a function home which is defined in views.py file.
views.py file
 

Python

from .models import Post

from .forms import PostForm

from .import views

from django.shortcuts import HttpResponse, render, redirect

def home(request):

    if request.method =='POST'

        details = PostForm(request.POST)

        if details.is_valid(): 

            post = details.save(commit = False)

            post.save() 

            return HttpResponse("data submitted successfully")

        else:

            return render(request, "home.html", {'form':details}) 

    else:

        form = PostForm(None)  

        return render(request, 'home.html', {'form':form})

home.html template file 
 

html

{% load bootstrap3 %}

{% bootstrap_messages %}

<!DOCTYPE html>

<html lang="en">

<head >

    <title>Basic Form</title>

    <meta charset="utf-8" />

    <meta name="viewport" content="width=device-width, initial-scale=1, shrink-to-fit=no">

</script>

</script>

</head>

<body  style="padding-top: 60px;background-color: #f5f7f8 !important;">

    <div class="container">

    <div class="row">

        <div class="col-md-4 col-md-offset-4">

        <h2>Form</h2>

             <form action="" method="post"><input type='hidden'/>

             {%csrf_token %}

                {% bootstrap_form form %}

<!-This is the form variable which we are passing from the function

of home in views.py file. That's the beauty of Django we

don't need to write much codes in this it'll automatically pass

all the form details in here

->

              <div class="form-group">

                <button type="submit" class="btn btn-default ">

                  Submit

                </button>

                  </div>

                </form>

        </div>

    </div>

</div>

</body>

</html>

Opening http://localhost:8000/ in the browser shows the following,
 

If a form with a username of length less than 5 is submitted, it gives an error at the time of submission and similarly for the Post Text filled as well. The following image shows how the form behaves on submitting invalid form data.
 

Понравилась статья? Поделить с друзьями:
  • Import tensorflow as tf ошибка
  • Import tensorflow as tf error
  • Import telebot python ошибка
  • Import system python ошибка
  • Import sqlalchemy error