我正在使用Django和Backblaze B2建立照片查看和存储网站,并且我打算在其上上传很多大(文件大小)照片。
我的计划是使用缩略图进行照片浏览,这些缩略图需要自动生成。将照片上传到Django,然后Django再将照片上传到B2,然后再次下载照片,创建缩略图并将缩略图上传到B2。
对我来说,下载完整尺寸的图像步骤似乎不是必需的,因为该文件已经通过Django网络服务器上传了。 Django不能只是将上传的照片保存在内存(或临时本地存储)中,构建缩略图,然后再将全尺寸图片和缩略图上传到B2吗?
我正在使用以下代码在save()
模型的Photo
步骤中生成缩略图,并且可以正常工作。我只是在寻找一种无需再次下载完整尺寸图像即可使其更高效的方法。我曾考虑在PhotoForm
中使用覆盖的save()
方法来执行此操作,但是我找不到如何执行此操作的方法。我还包含了自定义B2Storage
类的代码。
如果有人可以给我一种方法,我应该采取这种做法,我将不胜感激。谢谢!
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()
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']
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
答案 0 :(得分:0)
与此同时,我为此提出了一个合理的解决方案。我正在使用照片文件和其他要与其存储的数据创建TempPhoto
对象。我将照片存储在运行Django的服务器上的temp_photos
文件夹中。然后,在创建TempPhoto
对象之后,我调用Celery任务,并向上传的浏览器返回成功响应。
与此同时,芹菜任务开始在后台运行,这将创建一个真实的Photo
对象,并使用问题中介绍的make_thumbnail(self)
方法生成所需的缩略图。然后将所有必需的内容上载到B2。使用有时很慢的Backblaze B2 API,每张照片最多可能需要1分钟的时间,但这并不重要,因为它现在在后台异步运行,独立于Django Web服务器。在此任务结束时,TempPhoto
对象将被删除,temp_photo
中本地存储的照片也将被删除。
我正在寻找一种将任务进度转发给最终用户的方法,但是现在这并不是太重要。