运行collectstatic时收到以下错误:
AttributeError: 'cStringIO.StringO' object has no attribute 'name'
一切正常,直到我尝试使用存储“AWS_IS_GZIPPED”设置gzip我的文件(请参阅下面的settings.py)。
我正在使用django-storages,它利用boto将我的静态文件发送到S3和django-compressor来压缩它们。
以下是我的错误的完整追溯:
Copying '/home/somewhere/current/something/shows/static/skin/jplayer.band.css'
Traceback (most recent call last):
File "./manage.py", line 14, in <module>
execute_manager(settings)
File "/home/somewhere/env/lib/python2.6/site-packages/django/core/management/__init__.py", line 438, in execute_manager
utility.execute()
File "/home/somewhere/env/lib/python2.6/site-packages/django/core/management/__init__.py", line 379, in execute
self.fetch_command(subcommand).run_from_argv(self.argv)
File "/home/somewhere/env/lib/python2.6/site-packages/django/core/management/base.py", line 191, in run_from_argv
self.execute(*args, **options.__dict__)
File "/home/somewhere/env/lib/python2.6/site-packages/django/core/management/base.py", line 220, in execute
output = self.handle(*args, **options)
File "/home/somewhere/env/lib/python2.6/site-packages/django/core/management/base.py", line 351, in handle
return self.handle_noargs(**options)
File "/home/somewhere/env/lib/python2.6/site-packages/django/contrib/staticfiles/management/commands/collectstatic.py", line 89, in handle_noargs
self.copy_file(path, prefixed_path, storage, **options)
File "/home/somewhere/env/lib/python2.6/site-packages/django/contrib/staticfiles/management/commands/collectstatic.py", line 202, in copy_file
self.storage.save(prefixed_path, source_file)
File "/home/somewhere/current/something/storage.py", line 28, in save
self.local_storage._save(filename, content)
File "/home/somewhere/env/lib/python2.6/site-packages/django/core/files/storage.py", line 190, in _save
for chunk in content.chunks():
File "/home/somewhere/env/lib/python2.6/site-packages/django/core/files/base.py", line 65, in chunks
counter = self.size
File "/home/somewhere/env/lib/python2.6/site-packages/django/core/files/base.py", line 39, in _get_size
elif os.path.exists(self.file.name):
AttributeError: 'cStringIO.StringO' object has no attribute 'name'
这是我的storage.py
class CachedS3BotoStorage(S3BotoStorage):
def __init__(self, *args, **kwargs):
super(CachedS3BotoStorage, self).__init__(*args, **kwargs)
self.local_storage = get_storage_class('compressor.storage.CompressorFileStorage')()
def save(self, filename, content):
filename = super(CachedS3BotoStorage, self).save(filename, content)
self.local_storage._save(filename, content)
return filename
我的settings.py
中也有以下内容DEFAULT_FILE_STORAGE = 'storages.backends.s3boto.S3BotoStorage'
AWS_ACCESS_KEY_ID = 'something'
AWS_SECRET_ACCESS_KEY = 'something_else'
AWS_STORAGE_BUCKET_NAME = 'something'
AWS_S3_CUSTOM_DOMAIN = 'static.example.com' #without this compressor points to my s3 bucket, I serve from CloudFront
AWS_IS_GZIPPED = True #If this is false, everything works, but no gzipping!
AWS_S3_SECURE_URLS = False #turns off https for static files (necessary)
import datetime
future = datetime.datetime.now() #+ datetime.timedelta(days=364)
AWS_HEADERS = {
'Expires': future.strftime('%a, %d %b %Y %H:%M:%S GMT'),
'Cache-Control': 'max-age=60, public' #'max-age=31536000, public'
}
STATICFILES_STORAGE = 'something.storage.CachedS3BotoStorage'
from S3 import CallingFormat
AWS_CALLING_FORMAT = CallingFormat.SUBDOMAIN
COMPRESS_ENABLED = True #can remove when ready for production
COMPRESS_OFFLINE = True
COMPRESS_URL = 'http://static.example.com/'
COMPRESS_STORAGE = 'something.storage.GzipCompressorStorage'
COMPRESS_OUTPUT_DIR = 'compressed_static'
COMPRESS_ROOT = '/home/somewhere/static'
非常感谢任何帮助。在树林里有点迷失。
答案 0 :(得分:3)
答案 1 :(得分:0)
我用这种方式解决了这个问题:
问题是使用相同的[参数内容]来保存本地并上传到S3。因为当我们将参数内容传递给super(CachedS3BotoStorage,self).save(name,content)时,内部参数 content 被修改,所以当传递给下一个.save _self.local_storage时。保存(名称,内容)由于conf_AWS_IS_GZIPPED,他在其他州,压缩内容内的文件。 要解决此问题,只需创建一个内容参数副本。
在settings.py中
STATICFILES_STORAGE = 'mypackage.s3utils.CachedS3BotoStorage'
在mypackage.s3utils.py
中from storages.backends.s3boto import S3BotoStorage
from compressor.storage import CompressorFileStorage
from django.core.files.storage import get_storage_class
import copy
class CachedS3BotoStorage(S3BotoStorage):
"""
S3 storage backend that saves the files locally, too.
"""
location = 'static'
def __init__(self, *args, **kwargs):
super(CachedS3BotoStorage, self).__init__(*args, **kwargs)
self.local_storage = get_storage_class(
"compressor.storage.CompressorFileStorage")()
def url(self, name):
"""
Fix the problem of dont show the natives images django admin
"""
url = super(CachedS3BotoStorage, self).url(name)
if name.endswith('/') and not url.endswith('/'):
url += '/'
return url
def save(self, name, content):
content2 = copy.copy(content) #-> THE SECRET IS HERE
name = super(CachedS3BotoStorage, self).save(name, content)
self.local_storage._save(name, content2) #-> AND HERE
# print id(content)
# print id(content2)
return name
def get_available_name(self, name):
if self.exists(name):
self.delete(name)
return name