django-storages 및 Amazon S3를 사용하지만 정적 파일 및 미디어 파일에 대해 다른 폴더를 사용하여 Django 프로젝트를 설정하는 방법은 무엇입니까?


92

앱 정적 파일 ( STATIC_ROOT) 및 사용자 업로드 파일 ( MEDIA_ROOT) 을 저장하기 위해 서버 파일 시스템을 사용하는 Django 프로젝트를 구성하고 있습니다.

이제 모든 콘텐츠를 Amazon의 S3에 호스팅해야하므로이를위한 버킷을 만들었습니다. 스토리지 백엔드 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'

그럼, 문제가 생겼 다음을 MEDIA_ROOTSTATIC_ROOT버킷의 루트가 정적 파일과 사용자 업로드 경로를 모두 포함하므로, 버킷 내에서 사용되지 않습니다.

따라서 다음을 설정할 수 있습니다.

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

템플릿에서 이러한 설정을 사용하지만 .NET을 사용하여 S3에 저장할 때 정적 / 미디어 파일의 구분이 없습니다 django-storages.

어떻게 할 수 있습니까?

감사!


8
버킷의 이름 ( AWS_STORAGE_BUCKET_NAME) 을 지정하는 설정이 하나 뿐이고에 지정된 클래스의 인스턴스 STATICFILES_STORAGE가 인스턴스화 될 때 사용되는 설정이기 때문 입니다.
Armando Pérez Marqués

답변:


126

매우 유사하지만 다음이 작동해야하며 Mandx의 방법보다 간단해야한다고 생각합니다.

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'

(실제로 테스트 한) 다르지만 관련된 예제는 여기example_파일 에서 볼 수 있습니다 .


1
내 버전보다 확실히 더 간단하고 좋습니다. 나는 이것을 테스트하지 않았지만 이것이 효과가 있다고 생각합니다. 감사! 또한 django-s3storage repo를 확인하고 있는데 프로젝트가 S3를 독점적으로 사용하는 경우 매우 가벼운 솔루션 인 것 같습니다.
Armando Pérez Marqués

1
그리고 패키징에 더 관심이 있다면 django-s3-folder-storage를 확인하십시오 . 방금 찾았습니다.이 솔루션이 똑같은지 미리 포장되어 있는지 알 수 없습니다.
Armando Pérez Marqués

4
이것은 나에게 작동하지 않으며 미디어 파일이 s3 버킷의 /에 업로드됩니다. 위치 설정이 존중되지 않는 것 같습니다. 장고 - 스토리지 == 1.1.6, 장고 - 확장 == 1.1.1, 1.4 장고
나단 켈러

3
그것은 나에게는 별도의 버킷을 가지고 더 많은 감각을 만들어 내 솔루션이처럼 보이는 결국 그래서 내 설정 모듈의 I는 설정의 외부에있는 같이하지 gist.github.com/antonagestam/6075199
antonagestam

1
이 솔루션은 내가 말할 수있는 작동하지 않습니다. 이것은 접근 방식이어야합니다 : gist.github.com/defrex/82680e858281d3d3e6e4
defrex

8

현재이 코드를 별도의 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()의 "고정 된"버전을 사용하기 위해 private 메서드 를 재정의해야했습니다 .safe_join()SuspiciousOperation

나는 고려를 위해 이것을 게시하고 있습니다. 누구나 더 나은 대답을 주거나 이것을 개선 할 수 있다면 매우 환영받을 것입니다.


7

파일 : 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


이 파일 이름 일 경우 storages.py대신 custom_storages.py당신을 사용하는 것이 좋습니다from __future__ import absolute_import
아론 McMillin

2

대답은 매우 간단하고 기본적으로 수행됩니다. 이것은 Django 1.6.5 및 Boto 2.28.0을 사용하는 AWS Elastic Beanstalk에서 저에게 효과적입니다.

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 키는 컨테이너 구성 파일에서 전달되며 전혀 설정 하지 않았 STATIC_ROOT거나 STATIC_URL설정 하지 않았습니다 . 또한 s3utils.py파일이 필요하지 않습니다 . 이러한 세부 정보는 스토리지 시스템에서 자동으로 처리됩니다. 여기서 트릭은 템플릿에서이 알 수없는 경로를 정확하고 동적으로 참조해야한다는 것입니다. 예를 들면 :

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

이것이 .NET에서 로컬 (배포 전)에있는 파비콘을 해결하는 방법입니다 ~/Projects/my_app/project/my_app/static/img/favicon.ico.

물론 local_settings.py개발 환경에서 로컬로이 항목에 액세스하기위한 별도의 파일이 있으며 STATIC 및 MEDIA 설정이 있습니다. 이 솔루션을 찾기 위해 많은 실험과 독서를해야했으며 오류없이 일관되게 작동합니다.

정적 및 루트 분리가 필요하다는 것을 알고 있으며 버킷을 하나만 제공 할 수 있다는 점을 고려하면이 방법은 로컬 환경의 모든 폴더를 가져와 ~/Projects/my_app/project/my_app/static/버킷 루트 (예 : S3bucket / img /)에 폴더를 생성 한다는 점을 지적합니다 . 위의 예에서와 같이). 따라서 파일이 분리됩니다. 예를 들어 media폴더에 static폴더가 있고 다음과 같이 템플릿을 통해 액세스 할 수 있습니다 .

{% static "media/" %}

이게 도움이 되길 바란다. 저는 답을 찾기 위해 여기에 왔고 스토리지 시스템을 확장하는 것보다 더 간단한 솔루션을 찾기 위해 조금 더 힘을 냈습니다. 대신 Boto의 의도 된 사용에 대한 문서를 읽었고 필요한 것 중 많은 부분이 기본적으로 내장되어 있음을 발견했습니다. 건배!


당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.