如何设置Django的项目与Django的存储和亚马逊S3,但与静态文件和媒体文件不同的文件夹?

我正在configuration一个使用服务器文件系统来存储应用程序静态文件( STATIC_ROOT )和用户上传文件( MEDIA_ROOT )的Django项目。

我现在需要在亚马逊的S3上托pipe所有这些内容,所以我为此创build了一个存储桶。 使用django-storagesboto存储后端,我设法上传收集的静态到S3存储桶:

 MEDIA_ROOT = '/media/' STATIC_ROOT = '/static/' DEFAULT_FILE_STORAGE = 'storages.backends.s3boto.S3BotoStorage' AWS_ACCESS_KEY_ID = 'KEY_ID...' AWS_SECRET_ACCESS_KEY = 'ACCESS_KEY...' AWS_STORAGE_BUCKET_NAME = 'bucket-name' STATICFILES_STORAGE = 'storages.backends.s3boto.S3BotoStorage' 

然后,我遇到了一个问题:在STATIC_ROOT中没有使用MEDIA_ROOTSTATIC_ROOT ,所以bucket root同时包含静态文件和用户上传的path。

那么我可以设置:

 S3_URL = 'http://s3.amazonaws.com/%s' % AWS_STORAGE_BUCKET_NAME STATIC_URL = S3_URL + STATIC_ROOT MEDIA_URL = 'S3_URL + MEDIA_ROOT 

并在模板中使用这些设置,但是在使用django-storages存储在S3中时,没有静态/媒体文件的区别。

如何做到这一点?

谢谢!

我认为以下方法应该可行,比Mandx的方法简单,尽pipe它非常相似:

创build一个s3utils.py文件:

 from storages.backends.s3boto import S3BotoStorage StaticRootS3BotoStorage = lambda: S3BotoStorage(location='static') MediaRootS3BotoStorage = lambda: S3BotoStorage(location='media') 

然后在你的settings.py

 DEFAULT_FILE_STORAGE = 'myproject.s3utils.MediaRootS3BotoStorage' STATICFILES_STORAGE = 'myproject.s3utils.StaticRootS3BotoStorage' 

一个不同的但相关的例子(我已经testing过)可以在这里的两个example_文件中看到 。

我目前在一个单独的s3utils模块中使用这个代码:

 from django.core.exceptions import SuspiciousOperation from django.utils.encoding import force_unicode from storages.backends.s3boto import S3BotoStorage def safe_join(base, *paths): """ A version of django.utils._os.safe_join for S3 paths. Joins one or more path components to the base path component intelligently. Returns a normalized version of the final path. The final path must be located inside of the base path component (otherwise a ValueError is raised). Paths outside the base path indicate a possible security sensitive operation. """ from urlparse import urljoin base_path = force_unicode(base) paths = map(lambda p: force_unicode(p), paths) final_path = urljoin(base_path + ("/" if not base_path.endswith("/") else ""), *paths) # Ensure final_path starts with base_path and that the next character after # the final path is '/' (or nothing, in which case final_path must be # equal to base_path). base_path_len = len(base_path) - 1 if not final_path.startswith(base_path) \ or final_path[base_path_len:base_path_len + 1] not in ('', '/'): raise ValueError('the joined path is located outside of the base path' ' component') return final_path class StaticRootS3BotoStorage(S3BotoStorage): def __init__(self, *args, **kwargs): super(StaticRootS3BotoStorage, self).__init__(*args, **kwargs) self.location = kwargs.get('location', '') self.location = 'static/' + self.location.lstrip('/') def _normalize_name(self, name): try: return safe_join(self.location, name).lstrip('/') except ValueError: raise SuspiciousOperation("Attempted access to '%s' denied." % name) class MediaRootS3BotoStorage(S3BotoStorage): def __init__(self, *args, **kwargs): super(MediaRootS3BotoStorage, self).__init__(*args, **kwargs) self.location = kwargs.get('location', '') self.location = 'media/' + self.location.lstrip('/') def _normalize_name(self, name): try: return safe_join(self.location, name).lstrip('/') except ValueError: raise SuspiciousOperation("Attempted access to '%s' denied." % name) 

然后,在我的设置模块中:

 DEFAULT_FILE_STORAGE = 'myproyect.s3utils.MediaRootS3BotoStorage' STATICFILES_STORAGE = 'myproyect.s3utils.StaticRootS3BotoStorage' 

我需要重新定义_normalize_name()私有方法来使用safe_join()函数的“固定”版本,因为原始代码给了我合法path的SuspiciousOperationexception。

我发布这个考虑,如果任何人都可以给出更好的答案或改善这一个,这将是非常受欢迎的。


文件:PROJECT_NAME / custom_storages.py

 from django.conf import settings from storages.backends.s3boto import S3BotoStorage class StaticStorage(S3BotoStorage): location = settings.STATICFILES_LOCATION class MediaStorage(S3BotoStorage): location = settings.MEDIAFILES_LOCATION 

文件:PROJECT_NAME / settings.py

 STATICFILES_LOCATION = 'static' MEDIAFILES_LOCATION = 'media' if not DEBUG: STATICFILES_STORAGE = 'PROJECT_NAME.custom_storages.StaticStorage' DEFAULT_FILE_STORAGE = 'PROJECT_NAME.custom_storages.MediaStorage' AWS_ACCESS_KEY_ID = 'KEY_XXXXXXX' AWS_SECRET_ACCESS_KEY = 'SECRET_XXXXXXXXX' AWS_STORAGE_BUCKET_NAME = 'BUCKET_NAME' AWS_HEADERS = {'Cache-Control': 'max-age=86400',} AWS_QUERYSTRING_AUTH = False 

并运行: python manage.py collectstatic

我认为答案很简单,默认情况下完成。 这对我在AWS Elastic Beanstalk与Django 1.6.5和Boto 2.28.0:

 STATICFILES_FINDERS = ( 'django.contrib.staticfiles.finders.FileSystemFinder', 'django.contrib.staticfiles.finders.AppDirectoriesFinder', ) TEMPLATE_LOADERS = ( 'django.template.loaders.filesystem.Loader', 'django.template.loaders.app_directories.Loader', ) DEFAULT_FILE_STORAGE = 'storages.backends.s3boto.S3BotoStorage' STATICFILES_STORAGE = 'storages.backends.s3boto.S3BotoStorage' AWS_ACCESS_KEY_ID = os.environ['AWS_ACCESS_KEY_ID'] AWS_SECRET_ACCESS_KEY = os.environ['AWS_SECRET_KEY'] 

AWS密钥从容器configuration文件传入,我根本没有设置STATIC_ROOTSTATIC_URL 。 另外,不需要s3utils.py文件。 这些细节由存储系统自动处理。 这里的诀窍是我需要正确和dynamic地在我的模板中引用这个未知的path。 例如:

 <link rel="icon" href="{% static "img/favicon.ico" %}"> 

这就是我如何解决我在~/Projects/my_app/project/my_app/static/img/favicon.ico本地生活(预部署)的favicon。

当然,我有一个单独的local_settings.py文件在dev环境中本地访问这个东西,它有静态和媒体设置。 为了find这个解决scheme,我不得不做大量的实验和阅读工作,并且没有任何错误。

我知道你需要静态和根分离,并考虑到你只能提供一个桶,我会指出这种方法将我的本地环境中的所有文件夹~/Projects/my_app/project/my_app/static/并创build一个(例如:S3bucket / img /如上例)。 所以你得到文件的分离。 例如,您可以在static文件夹中有一个media文件夹,并通过以下模板进行访问:

 {% static "media/" %} 

我希望这有帮助。 我来到这里寻找答案,并且更加努力地find比扩展存储系统更简单的解决scheme。 相反,我阅读了有关Boto预期用途的文档,我发现我需要的很多是默认内置的。 干杯!