| 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 |
|---|