如何设置Django的项目与Django的存储和亚马逊S3,但与静态文件和媒体文件不同的文件夹?
我正在configuration一个使用服务器文件系统来存储应用程序静态文件( STATIC_ROOT
)和用户上传文件( MEDIA_ROOT
)的Django项目。
我现在需要在亚马逊的S3上托pipe所有这些内容,所以我为此创build了一个存储桶。 使用django-storages
与boto
存储后端,我设法上传收集的静态到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_ROOT
和STATIC_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的SuspiciousOperation
exception。
我发布这个考虑,如果任何人都可以给出更好的答案或改善这一个,这将是非常受欢迎的。
文件: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_ROOT
或STATIC_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预期用途的文档,我发现我需要的很多是默认内置的。 干杯!