Created
April 23, 2013 23:18
-
-
Save glarrain/5448253 to your computer and use it in GitHub Desktop.
Django model mixin to force Django to validate (i.e. call `full_clean`) before `save`
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
class ValidateModelMixin(object): | |
"""Make :meth:`save` call :meth:`full_clean`. | |
.. warning: | |
This should be the left-most mixin/super-class of a model. | |
Do you think Django models ``save`` method will validate all fields | |
(i.e. call ``full_clean``) before saving or any time at all? Wrong! | |
I discovered this awful truth when I couldn't understand why | |
a model object with an email field (without `blank=True`) could be | |
saved with an empty string as email address. | |
More info: | |
* "Why doesn't django's model.save() call full clean?" | |
http://stackoverflow.com/questions/4441539/ | |
* "Model docs imply that ModelForm will call Model.full_clean(), | |
but it won't." | |
https://code.djangoproject.com/ticket/13100 | |
""" | |
def save(self, *args, **kwargs): | |
"""Call :meth:`full_clean` before saving.""" | |
self.full_clean() | |
super(ValidateModelMixin, self).save(*args, **kwargs) |
I was looking at this as a real good model validation approach too, but indeed, query methods like update, select_for_update, bulk_create etc will all completely bypass this.
I was looking at this as a real good model validation approach too, but indeed, query methods like update, select_for_update, bulk_create etc will all completely bypass this.
Have you found an answer to this? On how to validate also in query methods like update, etc?
Is possible this validation?
Or whats the "final solution" for valid objects.
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
Be warned that this won't work for operations involving
.update()
, you should implement your constraint at the DB level