95

STATIC_ROOTアプリの静的ファイル ( ) とユーザーがアップロードしたファイル ( )を格納するためにサーバー ファイルシステムを使用していた Django プロジェクトを構成しています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_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

テンプレートでこれらの設定を使用しますが、django-storages.

これはどのように行うことができますか?

ありがとう!

4

6 に答える 6

127

非常に似ていますが、次の方法が機能し、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_の 2 つのファイルで確認できます。

于 2012-05-30T23:32:45.940 に答える
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

私はこれを検討のために投稿しています。誰かがより良い答えを与えるか、これを改善することができれば、それは大歓迎です。

于 2012-05-16T20:31:08.830 に答える
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_ROOTSTATIC_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 設定があります。この解決策を見つけるには、多くの実験と読み取りを行う必要がありましたが、エラーなしで一貫して機能します。

静的とルートの分離が必要であることを理解しています。提供できるバケットは 1 つだけであることを考えると、この方法では、ローカル環境のすべてのフォルダーが~/Projects/my_app/project/my_app/static/取得され、バケットのルート (つまり、S3bucket/img/) にフォルダーが作成されることを指摘します。上記の例のように)。したがって、ファイルの分離が得られます。たとえば、mediaフォルダー内にstaticフォルダーがあり、次のようにテンプレートを介してアクセスできます。

{% static "media/" %}

これが役立つことを願っています。私はその答えを探しにここに来て、ストレージ システムを拡張するよりも簡単な解決策を見つけるのに少し苦労しました。代わりに、Boto の使用目的に関するドキュメントを読んだところ、必要なものの多くがデフォルトで組み込まれていることがわかりました。乾杯!

于 2014-10-01T16:15:28.073 に答える