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 aField
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 Pythonfloat
or raise a
ValidationError
. -
The
validate()
method on aField
handles field-specific validation
that is not suitable for a validator. It takes a value that has been
coerced to a correct datatype and raisesValidationError
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 aField
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 aField
subclass is responsible for running
to_python()
,validate()
, andrun_validators()
in the correct
order and propagating their errors. If, at any time, any of the methods
raiseValidationError
, 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
inself.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
incleaned_data
because the general fieldclean()
method, above,
has already cleaned the data once).For example, if you wanted to validate that the contents of a
CharField
calledserialnumber
was unique,
clean_serialnumber()
would be the right place to do this. You don’t
need a specific field (it’s aCharField
), 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 fromcleaned_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 fieldA
is supplied, fieldB
must contain a valid email address”.
This method can return a completely different dictionary if it wishes, which
will be used as thecleaned_data
.Since the field validation methods have been run by the time
clean()
is
called, you also have access to the form’serrors
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
theclean()
method of aModelForm
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 ValidationError
s (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 code
s 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
превратит данные в Pythonfloat
или выдаст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 fieldclean()
, описанный выше, уже однажды очистил данные).Например, если вы хотите проверить, что содержимое
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()
значительно выигрывают от полнофункциональных 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'), ])
Использование валидации на практике¶
В предыдущих разделах объяснялось, как работает валидация в целом для форм. Поскольку иногда бывает проще понять, как работает каждая функция, здесь приведена серия небольших примеров, в которых используется каждая из предыдущих функций.
Использование валидаторов¶
Поля формы (и модели) 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.
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")
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")
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)
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',})
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)
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')
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)
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)
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.
When you hit submit, you’ll get an error as shown below:
Now we will try for a name in between 10 to 20 characters
Hit submit and check
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.