| 1 | from mimetypes import guess_type
|
|---|
| 2 | from StringIO import StringIO
|
|---|
| 3 | import urlparse
|
|---|
| 4 | import os
|
|---|
| 5 |
|
|---|
| 6 | from django.core.exceptions import ImproperlyConfigured
|
|---|
| 7 | from django.core.filestorage.base import Backend, RemoteFile
|
|---|
| 8 | from django.core.filestorage.filesystem import FileSystemBackend
|
|---|
| 9 | from django.utils.functional import curry
|
|---|
| 10 |
|
|---|
| 11 | ACCESS_KEY_NAME = 'AWS_ACCESS_KEY_ID'
|
|---|
| 12 | SECRET_KEY_NAME = 'AWS_SECRET_ACCESS_KEY'
|
|---|
| 13 |
|
|---|
| 14 | try:
|
|---|
| 15 | from _lib.amazon.S3 import AWSAuthConnection, QueryStringAuthGenerator, CallingFormat
|
|---|
| 16 | except ImportError:
|
|---|
| 17 | raise ImproperlyConfigured, "Could not load Amazon's S3 bindings.\
|
|---|
| 18 | \nSee http://developer.amazonwebservices.com/connect/entry.jspa?externalID=134"
|
|---|
| 19 |
|
|---|
| 20 | class S3Backend(Backend):
|
|---|
| 21 | """Amazon Simple Storage Service"""
|
|---|
| 22 |
|
|---|
| 23 | def __init__(self, bucket, access_key=None, secret_key=None,
|
|---|
| 24 | acl='public-read', calling_format=CallingFormat.REGULAR):
|
|---|
| 25 | self.bucket = bucket
|
|---|
| 26 | self.acl = acl
|
|---|
| 27 |
|
|---|
| 28 | if not access_key and not secret_key:
|
|---|
| 29 | access_key, secret_key = self._get_access_keys()
|
|---|
| 30 |
|
|---|
| 31 | self.connection = AWSAuthConnection(access_key, secret_key,
|
|---|
| 32 | calling_format=calling_format)
|
|---|
| 33 | self.generator = QueryStringAuthGenerator(access_key, secret_key,
|
|---|
| 34 | calling_format=calling_format, is_secure=False)
|
|---|
| 35 |
|
|---|
| 36 | def _get_access_keys(self):
|
|---|
| 37 | access_key = getattr(settings, ACCESS_KEY_NAME, None)
|
|---|
| 38 | secret_key = getattr(settings, SECRET_KEY_NAME, None)
|
|---|
| 39 | if (access_key or secret_key) and (not access_key or not secret_key):
|
|---|
| 40 | access_key = os.environ.get(ACCESS_KEY_NAME)
|
|---|
| 41 | secret_key = os.environ.get(SECRET_KEY_NAME)
|
|---|
| 42 |
|
|---|
| 43 | if access_key and secret_key:
|
|---|
| 44 | # Both were provided, so use them
|
|---|
| 45 | return access_key, secret_key
|
|---|
| 46 |
|
|---|
| 47 | return None, None
|
|---|
| 48 |
|
|---|
| 49 | def _get_connection(self):
|
|---|
| 50 | return AWSAuthConnection(*self._get_access_keys())
|
|---|
| 51 |
|
|---|
| 52 | def _put_file(self, filename, raw_contents):
|
|---|
| 53 | content_type = guess_type(filename)[0] or "application/x-octet-stream"
|
|---|
| 54 | headers = {'x-amz-acl': self.acl, 'Content-Type': content_type}
|
|---|
| 55 | response = self.connection.put(self.bucket, filename, raw_contents, headers)
|
|---|
| 56 |
|
|---|
| 57 | def get_absolute_url(self, filename):
|
|---|
| 58 | return self.generator.make_bare_url(self.bucket, filename)
|
|---|
| 59 |
|
|---|
| 60 | def get_filesize(self, filename):
|
|---|
| 61 | response = self.connection.make_request('HEAD', self.bucket, filename)
|
|---|
| 62 | return int(response.getheader('Content-Length'))
|
|---|
| 63 |
|
|---|
| 64 | def open_file(self, filename, mode='rb'):
|
|---|
| 65 | response = self.connection.get(self.bucket, filename)
|
|---|
| 66 | writer = curry(self._put_file, filename)
|
|---|
| 67 | return RemoteFile(self, response.object.data, mode, writer)
|
|---|
| 68 |
|
|---|
| 69 | def file_exists(self, filename):
|
|---|
| 70 | response = self.connection.make_request('HEAD', self.bucket, filename)
|
|---|
| 71 | return response.status == 200
|
|---|
| 72 |
|
|---|
| 73 | def save_file(self, filename, raw_contents):
|
|---|
| 74 | filename = self.get_available_filename(filename)
|
|---|
| 75 | self._put_file(filename, raw_contents)
|
|---|
| 76 | return filename
|
|---|
| 77 |
|
|---|
| 78 | def delete_file(self, filename):
|
|---|
| 79 | self.connection.delete(self.bucket, filename)
|
|---|
| 80 |
|
|---|
| 81 | def get_available_filename(self, filename):
|
|---|
| 82 | """ Overwrite existing file with the same name. """
|
|---|
| 83 | return filename
|
|---|