mirror of
https://github.com/pinry/pinry.git
synced 2025-11-13 16:45:41 +01:00
Only load Image once when creating all three Thumbnail sizes.
This commit is contained in:
@@ -35,8 +35,7 @@ class ImageManager(models.Manager):
|
|||||||
# a chance of getting Database into a inconsistent state when we
|
# a chance of getting Database into a inconsistent state when we
|
||||||
# try to create thumbnails one by one later
|
# try to create thumbnails one by one later
|
||||||
image = self.create(image=obj)
|
image = self.create(image=obj)
|
||||||
for size in settings.IMAGE_SIZES.keys():
|
Thumbnail.objects.get_or_create_at_sizes(image, settings.IMAGE_SIZES.keys())
|
||||||
Thumbnail.objects.get_or_create_at_size(image, size)
|
|
||||||
return image
|
return image
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -52,8 +52,7 @@ class ImageSerializer(serializers.ModelSerializer):
|
|||||||
|
|
||||||
def create(self, validated_data):
|
def create(self, validated_data):
|
||||||
image = super(ImageSerializer, self).create(validated_data)
|
image = super(ImageSerializer, self).create(validated_data)
|
||||||
for size in settings.IMAGE_SIZES:
|
Thumbnail.objects.get_or_create_at_sizes(image, settings.IMAGE_SIZES.keys())
|
||||||
Thumbnail.objects.get_or_create_at_size(image, size)
|
|
||||||
return image
|
return image
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -12,12 +12,6 @@ from .helpers import create_image, create_user, create_pin
|
|||||||
from core.models import Pin, Image
|
from core.models import Pin, Image
|
||||||
|
|
||||||
|
|
||||||
def filter_generator_for(size):
|
|
||||||
def wrapped_func(obj):
|
|
||||||
return Thumbnail.objects.get_or_create_at_size(obj, size)
|
|
||||||
return wrapped_func
|
|
||||||
|
|
||||||
|
|
||||||
def mock_requests_get(url, **kwargs):
|
def mock_requests_get(url, **kwargs):
|
||||||
response = mock.Mock(content=open('logo.png', 'rb').read())
|
response = mock.Mock(content=open('logo.png', 'rb').read())
|
||||||
return response
|
return response
|
||||||
|
|||||||
@@ -32,8 +32,7 @@ def create_tag(name):
|
|||||||
|
|
||||||
def create_image():
|
def create_image():
|
||||||
image = Image.objects.create(image=ImageFile(open(TEST_IMAGE_PATH, 'rb')))
|
image = Image.objects.create(image=ImageFile(open(TEST_IMAGE_PATH, 'rb')))
|
||||||
for size in settings.IMAGE_SIZES.keys():
|
Thumbnail.objects.get_or_create_at_sizes(image, settings.IMAGE_SIZES.keys())
|
||||||
Thumbnail.objects.get_or_create_at_size(image, size)
|
|
||||||
return image
|
return image
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -1,12 +1,10 @@
|
|||||||
import hashlib
|
import hashlib
|
||||||
import os.path
|
import os.path
|
||||||
from io import BytesIO
|
|
||||||
|
|
||||||
from django.db import models
|
from django.db import models
|
||||||
from django.core.files.uploadedfile import InMemoryUploadedFile
|
from django.core.files.uploadedfile import InMemoryUploadedFile
|
||||||
from django.core.urlresolvers import reverse
|
from django.core.urlresolvers import reverse
|
||||||
from django.dispatch import receiver
|
from django.dispatch import receiver
|
||||||
import PIL
|
|
||||||
|
|
||||||
try:
|
try:
|
||||||
from importlib import import_module
|
from importlib import import_module
|
||||||
@@ -67,37 +65,37 @@ class Image(models.Model):
|
|||||||
|
|
||||||
|
|
||||||
class ThumbnailManager(models.Manager):
|
class ThumbnailManager(models.Manager):
|
||||||
def get_or_create_at_size(self, image, size):
|
def get_or_create_at_sizes(self, image, sizes):
|
||||||
|
sizes_to_create = list(sizes)
|
||||||
|
sized = {}
|
||||||
|
for size in sizes:
|
||||||
if size not in IMAGE_SIZES:
|
if size not in IMAGE_SIZES:
|
||||||
raise ValueError("Received unknown size: %s" % size)
|
raise ValueError("Received unknown size: %s" % size)
|
||||||
|
|
||||||
try:
|
try:
|
||||||
thumbnail = image.get_by_size(size)
|
sized[size] = image.get_by_size(size)
|
||||||
except Thumbnail.DoesNotExist:
|
except Thumbnail.DoesNotExist:
|
||||||
img = utils.scale_and_crop(image.image, **IMAGE_SIZES[size])
|
pass
|
||||||
# save to memory
|
|
||||||
buf = BytesIO()
|
|
||||||
try:
|
|
||||||
img.save(buf, img.format, **img.info)
|
|
||||||
except IOError:
|
|
||||||
if img.info.get('progression'):
|
|
||||||
orig_MAXBLOCK = PIL.ImageFile.MAXBLOCK
|
|
||||||
temp_MAXBLOCK = 1048576
|
|
||||||
if orig_MAXBLOCK >= temp_MAXBLOCK:
|
|
||||||
raise
|
|
||||||
PIL.ImageFile.MAXBLOCK = temp_MAXBLOCK
|
|
||||||
try:
|
|
||||||
img.save(buf, img.format, **img.info)
|
|
||||||
finally:
|
|
||||||
PIL.ImageFile.MAXBLOCK = orig_MAXBLOCK
|
|
||||||
else:
|
else:
|
||||||
raise
|
sizes_to_create.remove(size)
|
||||||
|
|
||||||
|
if sizes_to_create:
|
||||||
|
bufs = [
|
||||||
|
utils.write_image_in_memory(img)
|
||||||
|
for img in utils.scale_and_crop_iter(
|
||||||
|
image.image,
|
||||||
|
[IMAGE_SIZES[size] for size in sizes_to_create])
|
||||||
|
]
|
||||||
|
for size, buf in zip(sizes_to_create, bufs):
|
||||||
# and save to storage
|
# and save to storage
|
||||||
original_dir, original_file = os.path.split(image.image.name)
|
original_dir, original_file = os.path.split(image.image.name)
|
||||||
thumb_file = InMemoryUploadedFile(buf, "image", original_file,
|
thumb_file = InMemoryUploadedFile(buf, "image", original_file,
|
||||||
None, buf.tell(), None)
|
None, buf.tell(), None)
|
||||||
thumbnail, created = image.thumbnail_set.get_or_create(
|
sized[size], created = image.thumbnail_set.get_or_create(
|
||||||
size=size, defaults={'image': thumb_file})
|
size=size, defaults={'image': thumb_file})
|
||||||
return thumbnail
|
|
||||||
|
# Make sure this is in the correct order
|
||||||
|
return [sized[size] for size in sizes]
|
||||||
|
|
||||||
|
|
||||||
class Thumbnail(models.Model):
|
class Thumbnail(models.Model):
|
||||||
|
|||||||
@@ -4,10 +4,10 @@ from django.test import TestCase
|
|||||||
from django.core.files.images import ImageFile
|
from django.core.files.images import ImageFile
|
||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
from django.utils.six import BytesIO
|
from django.utils.six import BytesIO
|
||||||
from django.core.urlresolvers import reverse
|
|
||||||
from django_images.models import Image, Thumbnail
|
from django_images.models import Image, Thumbnail
|
||||||
from django_images.templatetags.images import at_size
|
from django_images.templatetags.images import at_size
|
||||||
from django_images.utils import scale_and_crop
|
from django_images.utils import scale_and_crop_single
|
||||||
|
from PIL import Image as PILImage
|
||||||
|
|
||||||
|
|
||||||
class ImageModelTest(TestCase):
|
class ImageModelTest(TestCase):
|
||||||
@@ -20,7 +20,7 @@ class ImageModelTest(TestCase):
|
|||||||
|
|
||||||
def test_get_by_size(self):
|
def test_get_by_size(self):
|
||||||
size = list(settings.IMAGE_SIZES.keys())[0]
|
size = list(settings.IMAGE_SIZES.keys())[0]
|
||||||
Thumbnail.objects.get_or_create_at_size(self.image, size)
|
Thumbnail.objects.get_or_create_at_sizes(self.image, [size])
|
||||||
self.image.get_by_size(size)
|
self.image.get_by_size(size)
|
||||||
|
|
||||||
def test_get_absolute_url(self):
|
def test_get_absolute_url(self):
|
||||||
@@ -28,14 +28,9 @@ class ImageModelTest(TestCase):
|
|||||||
self.assertEqual(url, self.image.image.url)
|
self.assertEqual(url, self.image.image.url)
|
||||||
# For thumbnail
|
# For thumbnail
|
||||||
size = list(settings.IMAGE_SIZES.keys())[0]
|
size = list(settings.IMAGE_SIZES.keys())[0]
|
||||||
thumb = Thumbnail.objects.get_or_create_at_size(self.image, size)
|
thumb = Thumbnail.objects.get_or_create_at_sizes(self.image, [size])[0]
|
||||||
url = self.image.get_absolute_url(size)
|
url = self.image.get_absolute_url(size)
|
||||||
self.assertEqual(url, thumb.image.url)
|
self.assertEqual(url, thumb.image.url)
|
||||||
# Fallback on creation url
|
|
||||||
size = list(settings.IMAGE_SIZES.keys())[1]
|
|
||||||
url = self.image.get_absolute_url(size)
|
|
||||||
fallback_url = reverse('image-thumbnail', args=(self.image.id, size))
|
|
||||||
self.assertEqual(url, fallback_url)
|
|
||||||
|
|
||||||
|
|
||||||
class ThumbnailManagerModelTest(TestCase):
|
class ThumbnailManagerModelTest(TestCase):
|
||||||
@@ -48,17 +43,17 @@ class ThumbnailManagerModelTest(TestCase):
|
|||||||
self.size = list(settings.IMAGE_SIZES.keys())[0]
|
self.size = list(settings.IMAGE_SIZES.keys())[0]
|
||||||
|
|
||||||
def test_unknown_size(self):
|
def test_unknown_size(self):
|
||||||
self.assertRaises(ValueError, Thumbnail.objects.get_or_create_at_size,
|
self.assertRaises(ValueError, Thumbnail.objects.get_or_create_at_sizes,
|
||||||
self.image, 'foo')
|
self.image, ['foo'])
|
||||||
|
|
||||||
# TODO: Test the image object and data
|
# TODO: Test the image object and data
|
||||||
def test_create(self):
|
def test_create(self):
|
||||||
Thumbnail.objects.get_or_create_at_size(self.image, self.size)
|
Thumbnail.objects.get_or_create_at_sizes(self.image, [self.size])
|
||||||
self.assertEqual(self.image.thumbnail_set.count(), 1)
|
self.assertEqual(self.image.thumbnail_set.count(), 1)
|
||||||
|
|
||||||
def test_get(self):
|
def test_get(self):
|
||||||
thumb = Thumbnail.objects.get_or_create_at_size(self.image, self.size)
|
thumb = Thumbnail.objects.get_or_create_at_sizes(self.image, [self.size])[0]
|
||||||
thumb2 = Thumbnail.objects.get_or_create_at_size(self.image, self.size)
|
thumb2 = Thumbnail.objects.get_or_create_at_sizes(self.image, [self.size])[0]
|
||||||
self.assertEqual(thumb.id, thumb2.id)
|
self.assertEqual(thumb.id, thumb2.id)
|
||||||
|
|
||||||
|
|
||||||
@@ -70,7 +65,7 @@ class ThumbnailModelTest(TestCase):
|
|||||||
self.image = Image.objects.create(width=370, height=370,
|
self.image = Image.objects.create(width=370, height=370,
|
||||||
image=ImageFile(image_obj, '01.png'))
|
image=ImageFile(image_obj, '01.png'))
|
||||||
size = list(settings.IMAGE_SIZES.keys())[0]
|
size = list(settings.IMAGE_SIZES.keys())[0]
|
||||||
self.thumb = Thumbnail.objects.get_or_create_at_size(self.image, size)
|
self.thumb = Thumbnail.objects.get_or_create_at_sizes(self.image, [size])[0]
|
||||||
|
|
||||||
def test_get_absolute_url(self):
|
def test_get_absolute_url(self):
|
||||||
url = self.thumb.get_absolute_url()
|
url = self.thumb.get_absolute_url()
|
||||||
@@ -85,11 +80,11 @@ class PostSaveSignalOriginalChangedTestCase(TestCase):
|
|||||||
self.image = Image.objects.create(width=370, height=370,
|
self.image = Image.objects.create(width=370, height=370,
|
||||||
image=ImageFile(image_obj, '01.png'))
|
image=ImageFile(image_obj, '01.png'))
|
||||||
size = list(settings.IMAGE_SIZES.keys())[0]
|
size = list(settings.IMAGE_SIZES.keys())[0]
|
||||||
self.thumb = Thumbnail.objects.get_or_create_at_size(self.image, size)
|
self.thumb = Thumbnail.objects.get_or_create_at_sizes(self.image, [size])[0]
|
||||||
|
|
||||||
def test_post_save_signal_original_changed(self):
|
def test_post_save_signal_original_changed(self):
|
||||||
size = list(settings.IMAGE_SIZES.keys())[0]
|
size = list(settings.IMAGE_SIZES.keys())[0]
|
||||||
Thumbnail.objects.get_or_create_at_size(self.image, size)
|
Thumbnail.objects.get_or_create_at_sizes(self.image, [size])
|
||||||
self.image.delete()
|
self.image.delete()
|
||||||
self.assertFalse(Thumbnail.objects.exists())
|
self.assertFalse(Thumbnail.objects.exists())
|
||||||
|
|
||||||
@@ -102,7 +97,7 @@ class PostDeleteSignalDeleteImageFileTest(TestCase):
|
|||||||
self.image = Image.objects.create(width=370, height=370,
|
self.image = Image.objects.create(width=370, height=370,
|
||||||
image=ImageFile(image_obj, '01.png'))
|
image=ImageFile(image_obj, '01.png'))
|
||||||
size = list(settings.IMAGE_SIZES.keys())[0]
|
size = list(settings.IMAGE_SIZES.keys())[0]
|
||||||
self.thumb = Thumbnail.objects.get_or_create_at_size(self.image, size)
|
self.thumb = Thumbnail.objects.get_or_create_at_sizes(self.image, [size])[0]
|
||||||
|
|
||||||
@mock.patch('django_images.models.IMAGE_AUTO_DELETE', True)
|
@mock.patch('django_images.models.IMAGE_AUTO_DELETE', True)
|
||||||
def test_post_delete_signal_delete_image_files_enabled(self):
|
def test_post_delete_signal_delete_image_files_enabled(self):
|
||||||
@@ -136,7 +131,7 @@ class AtSizeTemplateTagTest(TestCase):
|
|||||||
self.image = Image.objects.create(width=370, height=370,
|
self.image = Image.objects.create(width=370, height=370,
|
||||||
image=ImageFile(image_obj, '01.png'))
|
image=ImageFile(image_obj, '01.png'))
|
||||||
size = list(settings.IMAGE_SIZES.keys())[0]
|
size = list(settings.IMAGE_SIZES.keys())[0]
|
||||||
self.thumb = Thumbnail.objects.get_or_create_at_size(self.image, size)
|
self.thumb = Thumbnail.objects.get_or_create_at_sizes(self.image, [size])[0]
|
||||||
|
|
||||||
def test_at_size(self):
|
def test_at_size(self):
|
||||||
size = list(settings.IMAGE_SIZES.keys())[0]
|
size = list(settings.IMAGE_SIZES.keys())[0]
|
||||||
@@ -144,65 +139,37 @@ class AtSizeTemplateTagTest(TestCase):
|
|||||||
self.assertEqual(url, self.thumb.image.url)
|
self.assertEqual(url, self.thumb.image.url)
|
||||||
|
|
||||||
|
|
||||||
class ThumbnailViewTest(TestCase):
|
|
||||||
def setUp(self):
|
|
||||||
image_obj = BytesIO()
|
|
||||||
qrcode_obj = qrcode.make('https://mirumee.com/')
|
|
||||||
qrcode_obj.save(image_obj)
|
|
||||||
self.image = Image.objects.create(width=370, height=370,
|
|
||||||
image=ImageFile(image_obj, '01.png'))
|
|
||||||
self.size = list(settings.IMAGE_SIZES.keys())[0]
|
|
||||||
self.thumb = Thumbnail.objects.get_or_create_at_size(self.image, self.size)
|
|
||||||
|
|
||||||
def test_redirect(self):
|
|
||||||
url = reverse('image-thumbnail', args=[self.image.id, self.size])
|
|
||||||
response = self.client.get(url)
|
|
||||||
self.assertNotEqual(url, self.thumb.image.url)
|
|
||||||
self.assertEqual(response.status_code, 302)
|
|
||||||
self.assertEqual(response.url, self.thumb.image.url)
|
|
||||||
|
|
||||||
def test_not_found(self):
|
|
||||||
url = reverse('image-thumbnail', args=['42', self.size])
|
|
||||||
response = self.client.get(url)
|
|
||||||
self.assertEqual(response.status_code, 404)
|
|
||||||
|
|
||||||
def test_size_not_found(self):
|
|
||||||
url = reverse('image-thumbnail', args=[self.image.id, '42'])
|
|
||||||
response = self.client.get(url)
|
|
||||||
self.assertEqual(response.status_code, 404)
|
|
||||||
|
|
||||||
|
|
||||||
class UtilsScaleAndDropTest(TestCase):
|
class UtilsScaleAndDropTest(TestCase):
|
||||||
def setUp(self):
|
def setUp(self):
|
||||||
image_obj = BytesIO()
|
image_obj = BytesIO()
|
||||||
qrcode_obj = qrcode.make('https://mirumee.com/')
|
qrcode_obj = qrcode.make('https://mirumee.com/')
|
||||||
qrcode_obj.save(image_obj)
|
qrcode_obj.save(image_obj)
|
||||||
self.imagefile = ImageFile(image_obj, '01.png')
|
self.image = PILImage.open(image_obj)
|
||||||
|
|
||||||
def test_change_size(self):
|
def test_change_size(self):
|
||||||
new_size = (10, 10)
|
new_size = (10, 10)
|
||||||
image = scale_and_crop(self.imagefile, new_size)
|
image = scale_and_crop_single(self.image, new_size)
|
||||||
self.assertEqual(new_size, image.im.size)
|
self.assertEqual(new_size, image.size)
|
||||||
|
|
||||||
def test_crop(self):
|
def test_crop(self):
|
||||||
new_size = (10, 10)
|
new_size = (10, 10)
|
||||||
image = scale_and_crop(self.imagefile, new_size, crop=True)
|
image = scale_and_crop_single(self.image, new_size, crop=True)
|
||||||
self.assertEqual(new_size, image.im.size)
|
self.assertEqual(new_size, image.size)
|
||||||
|
|
||||||
def test_disabled_upscale(self):
|
def test_disabled_upscale(self):
|
||||||
image = scale_and_crop(self.imagefile, (740, 740), upscale=False)
|
image = scale_and_crop_single(self.image, (740, 740), upscale=False)
|
||||||
self.assertLess(image.im.size[0], 371)
|
self.assertLess(image.size[0], 371)
|
||||||
self.assertLess(image.im.size[1], 371)
|
self.assertLess(image.size[1], 371)
|
||||||
|
|
||||||
def test_enaabled_upscale(self):
|
def test_enaabled_upscale(self):
|
||||||
image = scale_and_crop(self.imagefile, (740, 740), upscale=True)
|
image = scale_and_crop_single(self.image, (740, 740), upscale=True)
|
||||||
self.assertGreater(image.im.size[0], 371)
|
self.assertGreater(image.size[0], 371)
|
||||||
self.assertGreater(image.im.size[1], 371)
|
self.assertGreater(image.size[1], 371)
|
||||||
|
|
||||||
def test_not_change_quality(self):
|
def test_not_change_quality(self):
|
||||||
image = scale_and_crop(self.imagefile, (10, 10), quality=None)
|
image = scale_and_crop_single(self.image, (10, 10), quality=None)
|
||||||
self.assertEqual(image.info.get('quality'), None)
|
self.assertEqual(image.info.get('quality'), None)
|
||||||
|
|
||||||
def test_change_quality(self):
|
def test_change_quality(self):
|
||||||
image = scale_and_crop(self.imagefile, (10, 10), quality=50)
|
image = scale_and_crop_single(self.image, (10, 10), quality=50)
|
||||||
self.assertEqual(image.info.get('quality'), 50)
|
self.assertEqual(image.info.get('quality'), 50)
|
||||||
|
|||||||
@@ -1,6 +0,0 @@
|
|||||||
from django.conf.urls import url
|
|
||||||
from . import views
|
|
||||||
|
|
||||||
urlpatterns = [
|
|
||||||
url(r'^thumbnail/(?P<image_id>\d+)/(?P<size>[^/]+)/$', views.thumbnail, name='image-thumbnail'),
|
|
||||||
]
|
|
||||||
@@ -1,13 +1,45 @@
|
|||||||
|
from contextlib import contextmanager
|
||||||
|
from io import BytesIO
|
||||||
|
import PIL
|
||||||
from PIL import Image
|
from PIL import Image
|
||||||
|
|
||||||
|
|
||||||
|
@contextmanager
|
||||||
|
def open_django_file(fieldfile):
|
||||||
|
fieldfile.open()
|
||||||
|
try:
|
||||||
|
yield fieldfile
|
||||||
|
finally:
|
||||||
|
fieldfile.close()
|
||||||
|
|
||||||
|
|
||||||
|
def scale_and_crop_iter(image, options):
|
||||||
|
"""
|
||||||
|
Generator which will yield several variations on the input image.
|
||||||
|
Resize, crop and/or change quality of image.
|
||||||
|
|
||||||
|
:param image: Source image file
|
||||||
|
:param type: :class:`django.core.files.images.ImageFile
|
||||||
|
|
||||||
|
:param`options: List of option dictionaries, See scale_and_crop_single
|
||||||
|
argument names for available keys.
|
||||||
|
:type options: list of dict
|
||||||
|
"""
|
||||||
|
with open_django_file(image) as img:
|
||||||
|
im = Image.open(img)
|
||||||
|
im.load()
|
||||||
|
for opts in options:
|
||||||
|
# Use already-loaded file when cropping.
|
||||||
|
yield scale_and_crop_single(im, **opts)
|
||||||
|
|
||||||
|
|
||||||
# this neat function is based on easy-thumbnails
|
# this neat function is based on easy-thumbnails
|
||||||
def scale_and_crop(image, size, crop=False, upscale=False, quality=None):
|
def scale_and_crop_single(image, size, crop=False, upscale=False, quality=None):
|
||||||
"""
|
"""
|
||||||
Resize, crop and/or change quality of an image.
|
Resize, crop and/or change quality of an image.
|
||||||
|
|
||||||
:param image: Source image file
|
:param image: Source image file
|
||||||
:param type: :class:`django.core.files.images.ImageFile`
|
:param type: :class:`PIL.Image`
|
||||||
|
|
||||||
:param size: Size as width & height, zero as either means unrestricted
|
:param size: Size as width & height, zero as either means unrestricted
|
||||||
:type size: tuple of two int
|
:type size: tuple of two int
|
||||||
@@ -24,15 +56,7 @@ def scale_and_crop(image, size, crop=False, upscale=False, quality=None):
|
|||||||
:return: Handled image
|
:return: Handled image
|
||||||
:rtype: class:`PIL.Image`
|
:rtype: class:`PIL.Image`
|
||||||
"""
|
"""
|
||||||
# Open image and store format/metadata.
|
im = image
|
||||||
image.open()
|
|
||||||
im = Image.open(image)
|
|
||||||
im_format, im_info = im.format, im.info
|
|
||||||
if quality:
|
|
||||||
im_info['quality'] = quality
|
|
||||||
|
|
||||||
# Force PIL to load image data.
|
|
||||||
im.load()
|
|
||||||
|
|
||||||
source_x, source_y = [float(v) for v in im.size]
|
source_x, source_y = [float(v) for v in im.size]
|
||||||
target_x, target_y = [float(v) for v in size]
|
target_x, target_y = [float(v) for v in size]
|
||||||
@@ -68,6 +92,31 @@ def scale_and_crop(image, size, crop=False, upscale=False, quality=None):
|
|||||||
im = im.crop(box)
|
im = im.crop(box)
|
||||||
|
|
||||||
# Close image and replace format/metadata, as PIL blows this away.
|
# Close image and replace format/metadata, as PIL blows this away.
|
||||||
im.format, im.info = im_format, im_info
|
# We mutate the quality, but needs to passed into save() to actually
|
||||||
image.close()
|
# do anything.
|
||||||
|
info = image.info
|
||||||
|
if quality is not None:
|
||||||
|
info['quality'] = quality
|
||||||
|
im.format, im.info = image.format, info
|
||||||
return im
|
return im
|
||||||
|
|
||||||
|
|
||||||
|
def write_image_in_memory(img):
|
||||||
|
# save to memory
|
||||||
|
buf = BytesIO()
|
||||||
|
try:
|
||||||
|
img.save(buf, img.format, **img.info)
|
||||||
|
except IOError:
|
||||||
|
if img.info.get('progression'):
|
||||||
|
orig_MAXBLOCK = PIL.ImageFile.MAXBLOCK
|
||||||
|
temp_MAXBLOCK = 1048576
|
||||||
|
if orig_MAXBLOCK >= temp_MAXBLOCK:
|
||||||
|
raise
|
||||||
|
PIL.ImageFile.MAXBLOCK = temp_MAXBLOCK
|
||||||
|
try:
|
||||||
|
img.save(buf, img.format, **img.info)
|
||||||
|
finally:
|
||||||
|
PIL.ImageFile.MAXBLOCK = orig_MAXBLOCK
|
||||||
|
else:
|
||||||
|
raise
|
||||||
|
return buf
|
||||||
|
|||||||
@@ -1,13 +0,0 @@
|
|||||||
from django.http import HttpResponseNotFound
|
|
||||||
from django.shortcuts import get_object_or_404, redirect
|
|
||||||
|
|
||||||
from .models import Image, Thumbnail
|
|
||||||
from .settings import IMAGE_SIZES
|
|
||||||
|
|
||||||
|
|
||||||
def thumbnail(request, image_id, size):
|
|
||||||
image = get_object_or_404(Image, id=image_id)
|
|
||||||
if size not in IMAGE_SIZES:
|
|
||||||
return HttpResponseNotFound()
|
|
||||||
|
|
||||||
return redirect(Thumbnail.objects.get_or_create_at_size(image, size))
|
|
||||||
@@ -32,7 +32,3 @@ if settings.DEBUG:
|
|||||||
|
|
||||||
if settings.IS_TEST:
|
if settings.IS_TEST:
|
||||||
urlpatterns += staticfiles_urlpatterns()
|
urlpatterns += staticfiles_urlpatterns()
|
||||||
# For test running of django_images
|
|
||||||
urlpatterns += [
|
|
||||||
url(r'^__images/', include('django_images.urls')),
|
|
||||||
]
|
|
||||||
|
|||||||
Reference in New Issue
Block a user