'how to display models.full_clean() ValidationError in django admin?
https://docs.djangoproject.com/en/4.0/ref/models/instances/#validating-objects
from django.core.exceptions import ValidationError
try:
article.full_clean()
except ValidationError as e:
# Do something based on the errors contained in e.message_dict.
# Display them to a user, or handle them programmatically.
pass
There tell us can Display them to a user, how to display errors in Admin?
When I do nothing:
- When Settings.py Debug = True, it always render a ValidationError at /admin/xxx/xxx/xxx/change/ page.
- When Settings.py Debug = False, it always render a HTTP 500 page.
Some code in models.py:
def clean(self):
try:
if self.get_previous_state:
if self.get_previous_state.state_choice == self.state_choice and \
self.get_previous_state.state_location == self.state_location:
raise ValidationError({"state_choice": f"提交的状态与当前状态冲突({self.product_entity.rfidtag.EPC_bank})"},
params={'state_choice': self.state_choice})
except Exception as e: # except the RelatedObjectDoesNotExist Exception to ValidationError
raise ValidationError(e)
def save(self, force_insert=True, *args, **kwargs):
self.pk = None # force_insert 具有 pk unique 限制
self.previous_state = self.get_previous_state
self.full_clean()
super(ProductEntityState, self).save(force_insert=True, *args, **kwargs)
Solution 1:[1]
What is the context of that code? Django will automatically show the errors for you in the admin when creating/editing the object through the admin. You don't have to manually call full_clean()
as the auto-generated ModelForm from the admin will do so.
Solution 2:[2]
I fixed this bug by myself.
Because django admin just call formsets.all_valid()
to validtion in ModelAdmin._changeform_view()
.
ModelAdmin._changeform_view()
uses transaction.atomic()
.
When formsets.all_valid()
, the froms which will be deleted didn't be commit.
Now, the previous state is the instance which will be deleted.It will not raise the ValidationError
, the formsets.all_valid()
will return True
.
So ModelAdmin._changeform_view()
will call from.instance.save()
in foreach formsets.
This time, the previous state was be changed. So it will raise the ValidationError
.
But ModelAdmin._changeform_view()
didn't expect ValidationError
in ModelAdmin.save_related()
. it will expect by django\core\handlers\exception.py
and call response_for_exception()
.
I cover the ModelAdmin._create_formsets()
and fill the previous_state
in all formsets.instance
.
Notice: when you get form
, you should watch which model not call the model.clean()
, I use form.changed_data
to get instance.
Then, when ModelAdmin._create_formsets()
call formsets.all_valid()
. It will return False
, and raise ValidationError
in your page.
Sources
This article follows the attribution requirements of Stack Overflow and is licensed under CC BY-SA 3.0.
Source: Stack Overflow
Solution | Source |
---|---|
Solution 1 | Sebastián Bevc |
Solution 2 | 954 |