图片缩略图Django Backblaze B2

时间:2019-01-16 00:50:52

标签: python django

我正在使用Django和Backblaze B2建立照片查看和存储网站,并且我打算在其上上传很多大(文件大小)照片。

我的计划是使用缩略图进行照片浏览,这些缩略图需要自动生成。将照片上传到Django,然后Django再将照片上传到B2,然后再次下载照片,创建缩略图并将缩略图上传到B2。

对我来说,下载完整尺寸的图像步骤似乎不是必需的,因为该文件已经通过Django网络服务器上传了。 Django不能只是将上传的照片保存在内存(或临时本地存储)中,构建缩略图,然后再将全尺寸图片和缩略图上传到B2吗?

我正在使用以下代码在save()模型的Photo步骤中生成缩略图,并且可以正常工作。我只是在寻找一种无需再次下载完整尺寸图像即可使其更高效的方法。我曾考虑在PhotoForm中使用覆盖的save()方法来执行此操作,但是我找不到如何执行此操作的方法。我还包含了自定义B2Storage类的代码。

如果有人可以给我一种方法,我应该采取这种做法,我将不胜感激。谢谢!

models / photo.py

import os
from io import BytesIO

from PIL import Image
from django.core.files.base import ContentFile
from django.db import models

from fotoplatform.storage import B2Storage

THUMB_SIZE = (400, 400)


class Photo(models.Model):
    title = models.CharField(max_length=50)
    photo = models.ImageField(storage=B2Storage(), unique=True)
    thumbnail = models.ImageField(storage=B2Storage(), unique=True)

    def save(self, *args, **kwargs):
        if not self.thumbnail:
             self.make_thumbnail()
        super(Photo, self).save(*args, **kwargs)

    def make_thumbnail(self):
        try:
            image = Image.open(self.photo)
        except:
            raise Exception('Unable to open photo')

        image.thumbnail(THUMB_SIZE, Image.ANTIALIAS)

        thumb_name, thumb_extension = os.path.splitext(self.photo.name)
        thumb_extension = thumb_extension.lower()

        thumb_filename = thumb_name + '_thumb' + thumb_extension

        if thumb_extension in ['.jpg', '.jpeg']:
            FTYPE = 'JPEG'
        elif thumb_extension == '.gif':
            FTYPE = 'GIF'
        elif thumb_extension == '.png':
            FTYPE = 'PNG'
        else:
            raise Exception("Unknown extension")

        temp_thumb = BytesIO()
        image.save(temp_thumb, FTYPE)
        temp_thumb.seek(0)

        self.thumbnail.save(thumb_filename, ContentFile(temp_thumb.read()), save=True)
        temp_thumb.close()

forms.py

import logging

from django import forms

from fotoplatform.models import Photo


class PhotoForm(LoggingMixin, forms.ModelForm):
    photo = forms.FileField(widget=forms.ClearableFileInput(attrs={'multiple': True}))

    class Meta:
        model = Photo
        fields = ['title', 'photo']

storage.py

import hashlib
import logging
import os
import re

from b2blaze import B2
from django.core.files.storage import Storage
from django.utils.deconstruct import deconstructible

from dsbfotoplatform import settings

logger = logging.getLogger(__name__)

b2 = B2(key_id=settings.B2_KEY_ID, application_key=settings.B2_APPLICATION_KEY)
bucket = b2.buckets.get(bucket_id=settings.B2_BUCKET_ID)

@deconstructible
class B2Storage(Storage):

    def path(self, name):
        pass

    def delete(self, name):
        pass

    def exists(self, name):
        pass

    def listdir(self, path):
        pass

    def size(self, name):
        pass

    def get_accessed_time(self, name):
        pass

    def get_created_time(self, name):
        pass

    def get_modified_time(self, name):
        pass

    def _open(self, name, mode='rb'):
        file = bucket.files.get(file_name=name)
        return file.download()

    def _save(self, name, content):
        name = self.generate_filename(name)
        name = "photos/" + name
        bucket.files.upload(contents=content, file_name=name)
        return name

    def generate_filename(self, filename):
        filename, file_extension = os.path.splitext(filename)
        m = hashlib.md5()
        m.update(filename.encode("UTF-8"))
        return m.hexdigest() + file_extension

    def url(self, name):
        url = re.sub(r'b2api.*$', '', bucket.connector.download_url)
        url += "fotoplatform/" + name
        return url

1 个答案:

答案 0 :(得分:0)

与此同时,我为此提出了一个合理的解决方案。我正在使用照片文件和其他要与其存储的数据创建TempPhoto对象。我将照片存储在运行Django的服务器上的temp_photos文件夹中。然后,在创建TempPhoto对象之后,我调用Celery任务,并向上传的浏览器返回成功响应。

与此同时,芹菜任务开始在后台运行,这将创建一个真实的Photo对象,并使用问题中介绍的make_thumbnail(self)方法生成所需的缩略图。然后将所有必需的内容上载到B2。使用有时很慢的Backblaze B2 API,每张照片最多可能需要1分钟的时间,但这并不重要,因为它现在在后台异步运行,独立于Django Web服务器。在此任务结束时,TempPhoto对象将被删除,temp_photo中本地存储的照片也将被删除。

我正在寻找一种将任务进度转发给最终用户的方法,但是现在这并不是太重要。