Django在S3中存储上传的文件

时间:2014-08-04 21:10:41

标签: python django amazon-s3 boto

我有这个类使用Django REST框架将POST端点暴露给API使用者。

代码应该接收文件上传,然后将其上传到S3。文件正确上传到Django应用程序(file_obj.length返回实际文件大小),并在S3中创建对象。但是,S3中的文件大小为零。如果我记录file_obj.read()的返回值,那么它也是空的。

有什么问题?

from django.conf import settings

from rest_framework.views import APIView
from rest_framework.response import Response
from rest_framework.parsers import FileUploadParser
from boto.s3.connection import S3Connection
from boto.s3.key import Key

from .models import Upload
from .serializers import UploadSerializer


class UploadList(APIView):
    parser_classes = (FileUploadParser,)

    def post(self, request, format=None):
        file_obj = request.FILES['file']

        upload = Upload(user=request.user, file=file_obj)
        upload.save()

        conn = S3Connection(settings.AWS_ACCESS_KEY, settings.AWS_SECRET_KEY)
        k = Key(conn.get_bucket(settings.AWS_S3_BUCKET))
        k.key = 'upls/%s/%s.png' % (request.user.id, upload.key)
        k.set_contents_from_string(file_obj.read())

        serializer = UploadSerializer(upload)

        return Response(serializer.data, status=201)

2 个答案:

答案 0 :(得分:7)

是否有可能某些内容正在读取文件对象,也许是您的上传类保存方法,您需要回头查看?

file_obj.seek(0)

答案 1 :(得分:2)

您可以使用django存储

pip install django-storages

http://django-storages.readthedocs.org/en/latest/

在你的模特中,

def upload_image_to(instance, filename):
    import os
    from django.utils.timezone import now
    filename_base, filename_ext = os.path.splitext(filename)
    return 'posts/%s/%s' % (
        now().strftime("%Y%m%d"),
        instance.id
    )


image = models.ImageField(upload_to=upload_image_to, editable=True, null=True, blank=True)

在您的设置中,

DEFAULT_FILE_STORAGE = 'storages.backends.s3boto.S3BotoStorage'
AWS_S3_SECURE_URLS = False       # use http instead of https
AWS_QUERYSTRING_AUTH = False     # don't add complex authentication-related query parameters for requests

AWS_S3_ACCESS_KEY_ID = 'KEY'     # enter your access key id
AWS_S3_SECRET_ACCESS_KEY = 'KEY' # enter your secret access key
AWS_STORAGE_BUCKET_NAME = 'name.media'


INSTALLED_APPS = (
    ...
    'storages',

)