Validate image size in django admin
Solution 1
The right place to do this is during form validation.
A quick example (will edit/integrate with more info later):
from django.core.files.images import get_image_dimensions
from django.contrib import admin
from django import forms
class myForm(forms.ModelForm):
class Meta:
model = myModel
def clean_picture(self):
picture = self.cleaned_data.get("picture")
if not picture:
raise forms.ValidationError("No image!")
else:
w, h = get_image_dimensions(picture)
if w != 100:
raise forms.ValidationError("The image is %i pixel wide. It's supposed to be 100px" % w)
if h != 200:
raise forms.ValidationError("The image is %i pixel high. It's supposed to be 200px" % h)
return picture
class MyAdmin(admin.ModelAdmin):
form = myForm
admin.site.register(Example, MyAdmin)
Solution 2
use this function in your model file,
from django.core.exceptions import ValidationError
def validate_image(fieldfile_obj):
filesize = fieldfile_obj.file.size
megabyte_limit = 2.0
if filesize > megabyte_limit*1024*1024:
raise ValidationError("Max file size is %sMB" % str(megabyte_limit))
class Company(models.Model):
logo = models.ImageField("Logo", upload_to=upload_logo_to,validators=[validate_image], blank=True, null=True,help_text='Maximum file size allowed is 2Mb')
Solution 3
We can also use a class
with a __call__()
method to provide some parameters.
As suggested in Writing validators - Django Docs:
You can also use a class with a
__call__()
method for more complex or configurable validators.RegexValidator
, for example, uses this technique. If a class-based validator is used in thevalidators
model field option, you should make sure it is serializable by the migration framework by addingdeconstruct()
and__eq__()
methods.
Here is a working example:
try:
from collections.abc import Mapping
except ImportError:
from collections import Mapping
from django.utils.translation import ugettext_lazy as _
from django.utils.deconstruct import deconstructible
@deconstructible
class ImageValidator(object):
messages = {
"dimensions": _(
'Allowed dimensions are: %(width)s x %(height)s.'
),
"size": _(
"File is larger than > %(size)skB."
)
}
def __init__(self, size=None, width=None, height=None, messages=None):
self.size = size
self.width = width
self.height = height
if messages is not None and isinstance(messages, Mapping):
self.messages = messages
def __call__(self, value):
# _get_image_dimensions is a method of ImageFile
# https://docs.djangoproject.com/en/1.11/_modules/django/core/files/images/
if self.size is not None and value.size > self.size:
raise ValidationError(
self.messages['size'],
code='invalid_size',
params={
'size': float(self.size)/1024,
'value': value,
}
)
if (self.width is not None and self.height is not None and
(value.width != self.width or value.height != self.height)):
raise ValidationError(
self.messages['dimensions'],
code='invalid_dimensions',
params={
'width': self.width,
'height': self.height,
'value': value,
}
)
def __eq__(self, other):
return (
isinstance(other, self.__class__) and
self.size == other.size and
self.width == other.width and
self.height == other.height
)
And than in model:
class MyModel(models.Model):
...
banner = models.ImageField(
upload_to='uploads/', verbose_name=_("Banner"),
max_length=255, null=True, blank=True,
validators=[ImageValidator(size=256000, width=1140, height=425)],
help_text=_("Please use our recommended dimensions: 1140 x 425 PX, 250 KB MAX"))
Related videos on Youtube
Comments
-
Apreche over 2 years
I see a lot of people with Django apps that have image uploads are automatically resizing the images after they are uploaded. That is well and good for some cases, but I don't want to do this. Instead, I simply want to force the user to upload a file that is already the proper size.
I want to have an ImageField where I force the user to upload an image that is 100x200. If the image they upload is not exactly that size, I want the admin form to return as invalid. I would also like to be able to do the same thing for aspect ratios. I want to force the user to upload an image that is 16:9 and reject any upload that does not conform.
I already know how to get the width and height of the image, but I can't do that server-side until after the image is already uploaded, and the form is submitted successfully. How can I check this earlier, if possible?
-
nik_m about 6 yearsTake a look at django-vimage library. I have just created it and tries to solve such scenarios. Greetings :)
-
-
user2471801 over 14 yearsI was too slow, here's the related link: docs.djangoproject.com/en/dev/ref/forms/validation/…
-
cethegeek over 14 yearsWhile this solution is cool for reference, the OP said he wants to check the dimensions of the image BEFORE form submission. So it has to be something client side. Or am I misreading his last paragraph?
-
Apreche over 14 yearsI don't necessarily mean before form submission, as long as it's before the model is saved.
-
Agos over 14 yearsAnother possible technology could be a Java applet. I've seen quite a number of uploaders tackle the matter that way. It's also useful in case you actually want to allow big/plenty of uploads.
-
cethegeek over 14 yearsWell, then I think you found your winner! :-)
-
Nick T about 10 yearsEven if a pre-submission validator is used, you can never trust code run on the client so need to double-check.
-
bad_keypoints about 9 yearsget_image_dimension, highly useful ! Just apply it to request.FILES['image_field_name'] and bam! We get dimensions without saving the image already. Thanks.
-
Bryson over 8 yearsThis does not work. Referring to the file in this way from a validator, before the file has been saved, simply returns
*** FileNotFoundError: [Errno 2] No such file or directory: '/srv/www/media/80329e39-5c5c-4e79-8e32-51087ae00f6a'
. Perhaps it would work with files that Django does not save asInMemoryUploadedFile
objects, where they are in/tmp
or somewhere similar as an actual file. I have not tested that, however. -
Taiwotman over 6 yearsreally good solution.Thanks a lot. Looking at the
if not picture
. Can we apply this for the content-type of the image field? So instead ofif not picture
, it can beif not picture.content_type == "jpg"