无法使用boto通过Heroku将静态收集到s3-s3 bucket返回NoneType

时间:2013-01-09 04:41:49

标签: django heroku amazon-s3 amazon-cloudfront boto

这个问题涉及使用Heroku,django-storages(w / boto for s3)和CloudFront来提供静态内容的设置。

在过去的几个小时里,我一直在努力将我的静态文件成功加载到我的Heroku应用程序中。我已成功将Cloudfront挂钩到我的s3存储桶,看起来好像存储桶设置正确,但无论出于何种原因,AWS_STORAGE_BUCKET_NAME的值似乎都没有正确注册。

如果有人对如何调试这个有任何线索或想法,我会非常感激。我没办法。谢谢你的阅读。

settings.py(重要的东西):

try:
  from settings_local import *
except:
  import s3utils
DEBUG = False
#s3 stuff
DEFAULT_FILE_STORAGE = 's3utils.MediaRootS3BotoStorage'
STATICFILES_STORAGE = 's3utils.StaticRootS3BotoStorage'  
STATIC_URL = 'https://[domain].cloudfront.net/'
#use heroku postgres database
import dj_database_url
DATABASES['default'] =  dj_database_url.config()

s3utils.py

from storages.backends.s3boto import S3BotoStorage
from django.utils.functional import SimpleLazyObject
import os

AWS_ACCESS_KEY_ID = os.environ['AWS_ACCESS_KEY_ID']
AWS_SECRET_ACCESS_KEY = os.environ['AWS_SECRET_ACCESS_KEY']
AWS_STORAGE_BUCKET_NAME = 'static.[website].org'

StaticRootS3BotoStorage = lambda: S3BotoStorage(location='static')
MediaRootS3BotoStorage  = lambda: S3BotoStorage(location='media')

这是我在尝试通过'heroku run'或Procfile中收集静态时获得的回溯:

  Traceback (most recent call last):
  File "manage.py", line 10, in <module>
    execute_from_command_line(sys.argv)
  File "/app/.heroku/python/lib/python2.7/site-packages/django/core/management/__init__.py", line 443, in execute_from_command_line
    utility.execute()
  File "/app/.heroku/python/lib/python2.7/site-packages/django/core/management/__init__.py", line 382, in execute
    self.fetch_command(subcommand).run_from_argv(self.argv)
  File "/app/.heroku/python/lib/python2.7/site-packages/django/core/management/base.py", line 196, in run_from_argv
    self.execute(*args, **options.__dict__)
  File "/app/.heroku/python/lib/python2.7/site-packages/django/core/management/base.py", line 232, in execute
    output = self.handle(*args, **options)
  File "/app/.heroku/python/lib/python2.7/site-packages/django/core/management/base.py", line 371, in handle
    return self.handle_noargs(**options)
  File "/app/.heroku/python/lib/python2.7/site-packages/django/contrib/staticfiles/management/commands/collectstatic.py", line 163, in handle_noargs
    collected = self.collect()
  File "/app/.heroku/python/lib/python2.7/site-packages/django/contrib/staticfiles/management/commands/collectstatic.py", line 113, in collect
    handler(path, prefixed_path, storage)
  File "/app/.heroku/python/lib/python2.7/site-packages/django/contrib/staticfiles/management/commands/collectstatic.py", line 287, in copy_file
    if not self.delete_file(path, prefixed_path, source_storage):
  File "/app/.heroku/python/lib/python2.7/site-packages/django/contrib/staticfiles/management/commands/collectstatic.py", line 215, in delete_file
    if self.storage.exists(prefixed_path):
  File "/app/.heroku/python/lib/python2.7/site-packages/storages/backends/s3boto.py", line 284, in exists
    return k.exists()
  File "/app/.heroku/python/lib/python2.7/site-packages/boto/s3/key.py", line 399, in exists
    return bool(self.bucket.lookup(self.name))
  File "/app/.heroku/python/lib/python2.7/site-packages/boto/s3/bucket.py", line 148, in lookup
    return self.get_key(key_name, headers=headers)
  File "/app/.heroku/python/lib/python2.7/site-packages/boto/s3/bucket.py", line 181, in get_key
    query_args=query_args)
  File "/app/.heroku/python/lib/python2.7/site-packages/boto/s3/connection.py", line 458, in make_request
    auth_path = self.calling_format.build_auth_path(bucket, key)
  File "/app/.heroku/python/lib/python2.7/site-packages/boto/s3/connection.py", line 92, in build_auth_path
    path = '/' + bucket
TypeError: cannot concatenate 'str' and 'NoneType' objects

请注意,我已经省略了域名等,我在代码中实际上没有[域名]或[网站]。

1 个答案:

答案 0 :(得分:14)

您没有说from s3utils import *,因此永远不会将AWS_STORAGE_BUCKET_NAME导入设置模块。

S3BotoStorage会从环境变量中提取AWS_ACCESS_KEY_IDAWS_SECRET_ACCESS_KEY,但必须在AWS_STORAGE_BUCKET_NAME中设置settings.py。这似乎是一个奇怪的不一致,但我认为这是因为AWS_ACCESS_KEY_IDAWS_SECRET_ACCESS_KEY实际上是boto参数而AWS_STORAGE_BUCKET_NAME不是(boto将从env vars)。*

对s3utils的其他引用是明确的:

DEFAULT_FILE_STORAGE = 's3utils.MediaRootS3BotoStorage'
STATICFILES_STORAGE = 's3utils.StaticRootS3BotoStorage'

因此,唯一没有处理的设置是AWS_STORAGE_BUCKET_NAME,这会导致您的错误。

*我想看看django-storages接受来自env vars(12-Factor App, anyone?)的其他设置,并且我正在考虑打开一个问题/提交拉取请求。