Як налаштувати проект Django з django-сховищами та Amazon S3, але з різними папками для статичних файлів та медіафайлів?


92

Я налаштовую проект Django, який використовував серверну файлову систему для зберігання статичних файлів програм ( STATIC_ROOT) та завантажених користувачем файлів ( MEDIA_ROOT).

Зараз мені потрібно розмістити весь цей вміст на 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_ROOTі STATIC_ROOTне використовуються в сегменті, тому корінь сегмента містить як статичні файли, так і завантажені користувачами шляхи.

Тоді я міг встановити:

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

І використовуйте ці налаштування в шаблонах, але немає ніякої різниці між статичними / медіафайлами при зберіганні в S3 з django-storages.

Як це можна зробити?

Дякую!


8
Оскільки для вказівки імені сегмента ( AWS_STORAGE_BUCKET_NAME) існує лише один параметр , і це той, який використовується, коли екземпляр класу, зазначеного в STATICFILES_STORAGE, інстанціюється.
Армандо Перес Маркес,

Відповіді:


126

Я думаю, що наступне має працювати і бути простішим, ніж метод Мандкса, хоча він дуже схожий:

Створити 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 , здається дуже легким рішенням, якщо проект використовує виключно S3.
Армандо Перес Маркес

1
Якщо ви більше любите упаковку, перевірте django-s3-folder-storage . Я щойно знайшов його, не можу сказати, чи це саме таке рішення, але фасоване.
Армандо Перес Маркес,

4
Це не працює у мене, мультимедійні файли завантажуються в / з сегмента s3. Здається, налаштування місцезнаходження не дотримуються. django-storages == 1.1.6, django-extensions == 1.1.1, django = 1.4
Натан Келлер

3
Для нього мені було більше сенсу мати окремі сегменти, і мені не подобається мати конфігурацію поза моїм модулем налаштувань, тому моє рішення закінчилося виглядати так: 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()приватний метод, щоб використовувати "фіксовану" версію 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
Аарон Макміллін,

2

Я думаю, що відповідь досить проста і зроблена за замовчуванням. Це працює для мене на 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 передаються з конфігураційного файлу контейнера, і я взагалі не маю STATIC_ROOTабо STATIC_URLвстановлений. Крім того, немає необхідності у s3utils.pyфайлі. Ці деталі обробляються системою зберігання автоматично. Фокус у тому, що мені потрібно було правильно та динамічно посилатися на цей невідомий шлях у своїх шаблонах. Наприклад:

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

Ось як я звертаюся до свого фавікона, який живе локально (до розгортання) в ~/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 за призначенням і виявив, що багато з того, що мені потрібно, було вбудовано за замовчуванням. Ура!


Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.