Heroku& amp;亚马逊S3,所以请耐心等待。将我的Django应用程序上传到Heroku,并且用户媒体上传时出现问题。该模型如下:
#models.py
class Movie(models.Model):
title = models.CharField(max_length = 500)
poster = models.ImageField(upload_to = 'storages.backends.s3boto')
pub_date = models.DateTimeField(auto_now_add = True)
author = models.ForeignKey(User)
海报属性是上传图像的属性。我在本地运行良好,现在在Heroku上有一个错误。所以我添加了'storages.backends.s3boto',正如许多其他帖子告诉我的那样。 (不确定是否正确)。
我的Settings.py文件现在看起来像这样,有点混乱:
#settings.py
PROJECT_ROOT = os.path.abspath(os.path.dirname(__file__))
PROJECT_DIR = os.path.join(PROJECT_ROOT, '../qanda')
DEFAULT_FILE_STORAGE = 'storages.backends.s3boto.S3BotoStorage'
STATICFILES_STORAGE = 'storages.backends.s3boto.S3BotoStorage'
AWS_ACCESS_KEY_ID = '****************'
AWS_SECRET_ACCESS_KEY = '************'
AWS_STORAGE_BUCKET_NAME = 'mrt-assets'
AWS_PRELOAD_METADATA = True
MEDIA_ROOT = os.path.join(PROJECT_ROOT, 'qanda/media/movie_posters/)
MEDIA_URL = '/media'
STATIC_ROOT = os.path.join(PROJECT_ROOT, 'staticfiles')
STATIC_URL = 'https://mrt-assets.s3.amazonaws.com/static/'
STATICFILES_DIRS = (os.path.join(PROJECT_DIR, 'static'),)
我的存储桶被称为mrt-assets,其中有2个静态文件夹(css,js,图像和媒体。我现在不太担心静态文件,因为我已经硬编码了CSS / JS文件放入我的HTML文件*,但如何将我的用户上传的媒体(任何类型的图像)放入mrt-assets / media?
*虽然如果有人想要帮助STATIC文件也会很棒。但用户上传媒体更为紧迫。
编辑(根据Yuji的评论):尝试了很多选项,但都没有。我已经回去并删除了很多更改,现在这是我的设置
#settings.py
PROJECT_ROOT = os.path.abspath(os.path.dirname(__file__))
MEDIA_ROOT = 'http://s3.amazonaws.com/mrt-assets/media/'
MEDIA_URL = '/media/'
STATIC_ROOT = 'http://s3.amazonaws.com/mrt-assets/static/'
STATIC_URL = '/static/'
ADMIN_MEDIA_PREFIX = STATIC_URL + 'admin/'
TEMPLATE_DIRS = (os.path.join(PROJECT_ROOT, "templates"),)
#models.py
#same as before, but now have changed the poster directory
poster = models.ImageField().
不确定要做什么,需要将我的Heroku应用程序连接到S3,以便将用户媒体上传保存在那里。
现在已将我的S3 Bucket改为此
mrt-assets
static
css
js
images
media
(empty)
答案 0 :(得分:5)
将媒体上传到<bucket>/media
并将静态资源上传到<bucket>/static
的技巧是为两种资产类型创建两个不同的默认存储后端,或使用存储显式实例化模型字段对象采用location
参数。
from storages.backends.s3boto import S3BotoStorage
class Movie(models.Model):
title = models.CharField(max_length=500)
poster = models.ImageField(storage=S3BotoStorage(location='media'))
pub_date = models.DateTimeField(auto_now_add=True)
author = models.ForeignKey(User)
给S3BotoStorage
一个location
将为所有上传添加其路径前缀。
这与显式定义存储后端几乎相同
位置,但我们将使用settings.MEDIA_ROOT
和
settings.STATIC_ROOT
全局应用路径前缀。
# settings.py
STATIC_ROOT = '/static/'
MEDIA_ROOT = '/media/'
DEFAULT_FILE_STORAGE = 'app.storage.S3MediaStorage'
STATICFILES_STORAGE = 'app.storage.S3StaticStorage'
# app/storage.py
from django.conf import settings
from storages.backends.s3boto import S3BotoStorage
class S3MediaStorage(S3BotoStorage):
def __init__(self, **kwargs):
kwargs['location'] = kwargs.get('location',
settings.MEDIA_ROOT.replace('/', ''))
super(S3MediaStorage, self).__init__(**kwargs)
class S3StaticStorage(S3BotoStorage):
def __init__(self, **kwargs):
kwargs['location'] = kwargs.get('location',
settings.STATIC_ROOT.replace('/', ''))
super(S3StaticStorage, self).__init__(**kwargs)
您可以优化上述代码以利用 Heroku config vars 使它更便携:
# settings.py
import os
STATIC_ROOT = os.environ.get('STATIC_ROOT',
os.path.join(os.path.dirname(__file__), 'static'))
MEDIA_ROOT = os.environ.get('MEDIA_ROOT',
os.path.join(os.path.dirname(__file__), 'media'))
DEFAULT_FILE_STORAGE = os.environ.get('DEFAULT_FILE_STORAGE',
'django.core.files.storage.FileSystemStorage')
STATICFILES_STORAGE = os.environ.get('STATICFILES_STORAGE',
'django.contrib.staticfiles.storage.StaticFilesStorage')
将上述设置与.env
文件相结合,您可以使用
本地默认存储后端用于开发和测试以及何时
在Heroku上部署,您将自动切换到
分别为app.storage.S3MediaStorage
和app.storage.S3StaticStorage
:
# .env
STATIC_ROOT=static
MEDIA_ROOT=media
DEFAULT_FILE_STORAGE=app.storage.S3MediaStorage
STATICFILES_STORAGE=app.storage.S3StaticStorage