Skip to content

Instantly share code, notes, and snippets.

@insin
Created February 9, 2011 04:21
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save insin/817873 to your computer and use it in GitHub Desktop.
Save insin/817873 to your computer and use it in GitHub Desktop.
ImageURLField for Django forms
import urllib
from django import forms
from django.template.defaultfilters import filesizeformat
from django.utils.text import get_text_list
# Try to import PIL in either of the two ways it can end up installed
try:
from PIL import ImageFile as PILImageFile
except ImportError:
import ImageFile as PILImageFile
class ImageURLField(forms.URLField):
"""
A URL field specifically for images, which can validate details
about the filesize, dimensions and format of an image at a given
URL.
Specifying any of the following arguments will result in the
appropriate validation of image details, retrieved from the URL
specified in this field:
max/min_filesize
An integer specifying an image filesize limit, in bytes.
max/min_width
An integer specifying an image width limit, in pixels.
max/min_height
An integer specifying an image height limit, in pixels.
image_formats
A list of image formats to be accepted, specified as uppercase
strings.
For a list of valid image formats, see the "Image File Formats"
section of the `Python Imaging Library Handbook`_.
.. _`Python Imaging Library Handbook`: http://www.pythonware.com/library/pil/handbook/
"""
def __init__(self, max_filesize=None, min_filesize=None, max_width=None,
min_width=None, max_height=None, min_height=None, image_formats=None,
*args, **kwargs):
super(ImageURLField, self).__init__(*args, **kwargs)
self.max_filesize, self.min_filesize = max_filesize, min_filesize
self.max_width, self.min_width = max_width, min_width
self.max_height, self.min_height = max_height, min_height
self.image_formats = image_formats
self.validate_image = \
max_filesize is not None or min_filesize is not None or \
max_width is not None or min_width is not None or \
max_height is not None or min_height is not None or \
image_formats is not None
def validate(self, value):
super(ImageURLField, self).validate(value)
if value == '' or not self.validate_image:
return
try:
filesize, dimensions, format = self._get_image_details(value)
if dimensions is None or format is None:
raise forms.ValidationError(
'Could not retrieve image details from this URL.')
if self.max_filesize is not None and filesize > self.max_filesize:
raise forms.ValidationError(
'The image at this URL is %s large - it must be at most %s.' % (
filesizeformat(filesize), filesizeformat(self.max_filesize)))
if self.min_filesize is not None and filesize < self.min_filesize:
raise forms.ValidationError(
'The image at this URL is %s large - it must be at least %s.' % (
filesizeformat(filesize), filesizeformat(self.min_filesize)))
if self.max_width is not None and dimensions[0] > self.max_width:
raise forms.ValidationError(
'The image at this URL is %s pixels wide - it must be at most %s pixels.' % (
dimensions[0], self.max_width))
if self.min_width is not None and dimensions[0] < self.min_width:
raise forms.ValidationError(
'The image at this URL is %s pixels wide - it must be at least %s pixels.' % (
dimensions[0], self.min_width))
if self.max_height is not None and dimensions[1] > self.max_height:
raise forms.ValidationError(
'The image at this URL is %s pixels high - it must be at most %s pixels.' % (
dimensions[1], self.max_height))
if self.min_height is not None and dimensions[1] < self.min_height:
raise forms.ValidationError(
'The image at this URL is %s pixels high - it must be at least %s pixels.' % (
dimensions[1], self.min_height))
if self.image_formats is not None and format not in self.image_formats:
raise forms.ValidationError(
'The image at this URL is in %s format - %s %s.' % (
format,
len(self.image_formats) == 1 and 'the only accepted format is' or 'accepted formats are',
get_text_list(self.image_formats)))
except IOError:
raise forms.ValidationError('Could not load an image from this URL.')
return value
def _get_image_details(self, url):
"""
Retrieves details about the image accessible at the given URL,
returning a 3-tuple of (filesize, image dimensions (width,
height) and image format), or (filesize, ``None``, ``None``) if
image details could not be determined.
The Python Imaging Library is used to parse the image in chunks
in order to determine its dimension and format details without
having to load the entire image into memory.
Adapted from http://effbot.org/zone/pil-image-size.htm
"""
file = urllib.urlopen(url)
filesize = file.headers.get('content-length')
if filesize: filesize = int(filesize)
p = PILImageFile.Parser()
while 1:
data = file.read(1024)
if not data:
break
p.feed(data)
if p.image:
return filesize, p.image.size, p.image.format
break
file.close()
return filesize, None, None
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment