【问题标题】:Collectstatic error while deploying Django app to Heroku在将 Django 应用程序部署到 Heroku 时收集静态错误
【发布时间】:2016-08-08 12:47:22
【问题描述】:

我正在尝试将 Django 应用程序部署到 Heroku,它开始构建、下载和安装所有内容,但这就是我在收集静态文件时得到的结果

$ python manage.py collectstatic --noinput
remote:        Traceback (most recent call last):
remote:          File "manage.py", line 10, in <module>
remote:            execute_from_command_line(sys.argv)
remote:          File "/app/.heroku/python/lib/python2.7/site-packages/django/core/management/__init__.py", line 338, in execute_from_command_line
remote:            utility.execute()
remote:          File "/app/.heroku/python/lib/python2.7/site-packages/django/core/management/__init__.py", line 330, in execute
remote:            self.fetch_command(subcommand).run_from_argv(self.argv)
remote:          File "/app/.heroku/python/lib/python2.7/site-packages/django/core/management/base.py", line 390, in run_from_argv
remote:            self.execute(*args, **cmd_options)
remote:          File "/app/.heroku/python/lib/python2.7/site-packages/django/core/management/base.py", line 441, in execute
remote:            output = self.handle(*args, **options)
remote:          File "/app/.heroku/python/lib/python2.7/site-packages/django/contrib/staticfiles/management/commands/collectstatic.py", line 168, in handle
remote:            collected = self.collect()
remote:          File "/app/.heroku/python/lib/python2.7/site-packages/django/contrib/staticfiles/management/commands/collectstatic.py", line 98, in collect
remote:            for path, storage in finder.list(self.ignore_patterns):
remote:          File "/app/.heroku/python/lib/python2.7/site-packages/django/contrib/staticfiles/finders.py", line 112, in list
remote:            for path in utils.get_files(storage, ignore_patterns):
remote:          File "/app/.heroku/python/lib/python2.7/site-packages/django/contrib/staticfiles/utils.py", line 28, in get_files
remote:            directories, files = storage.listdir(location)
remote:          File "/app/.heroku/python/lib/python2.7/site-packages/django/core/files/storage.py", line 300, in listdir
remote:            for entry in os.listdir(path):
remote:        OSError: [Errno 2] No such file or directory: '/app/blogproject/static'
remote: 
remote:  !     Error while running '$ python manage.py collectstatic --noinput'.
remote:        See traceback above for details.
remote: 
remote:        You may need to update application code to resolve this error.
remote:        Or, you can disable collectstatic for this application:
remote: 
remote:           $ heroku config:set DISABLE_COLLECTSTATIC=1
remote: 
remote:        https://devcenter.heroku.com/articles/django-assets
remote: 
remote:  !     Push rejected, failed to compile Python app
remote: 
remote: Verifying deploy...
remote: 
remote: !   Push rejected to pin-a-voyage.

这是整个 settings.py 文件

# Build paths inside the project like this: os.path.join(BASE_DIR, ...)
import os
import dj_database_url

BASE_DIR = os.path.dirname(os.path.dirname(__file__))
PROJECT_ROOT = os.path.dirname(os.path.abspath(__file__))



# Quick-start development settings - unsuitable for production
# See https://docs.djangoproject.com/en/1.8/howto/deployment/checklist/

# SECURITY WARNING: keep the secret key used in production secret!
SECRET_KEY = '*********************'

# SECURITY WARNING: don't run with debug turned on in production!
DEBUG = True


# Application definition

INSTALLED_APPS = (
    'django.contrib.admin',
    'django.contrib.auth',
    'django.contrib.contenttypes',
    'django.contrib.sessions',
    'django.contrib.messages',
    'django.contrib.staticfiles',
    'blog',
    'custom_user',
    'django_markdown',
    'parsley',
)

#### AUTH ###

AUTH_USER_MODEL = 'custom_user.CustomUser'

AUTHENTICATION_BACKENDS = (
    'custom_user.backends.CustomUserAuth',
    'django.contrib.auth.backends.ModelBackend',
    # 'django.contrib.auth.backends.RemoteUserBackend',
)

#############

#### EMAIL ###

EMAIL_USE_TLS = True
EMAIL_BACKEND = 'django.core.mail.backends.smtp.EmailBackend'
EMAIL_HOST = 'smtp.gmail.com'
EMAIL_HOST_PASSWORD = '***' #my gmail password
EMAIL_HOST_USER = 'voyage.pin@gmail.com' #my gmail username
DEFAULT_FROM_EMAIL = 'voyage.pin@gmail.com'
SERVER_EMAIL = 'voyage.pin@gmail.com'
EMAIL_PORT = 587
DEFAULT_FROM_EMAIL = EMAIL_HOST_USER

##############

MIDDLEWARE_CLASSES = (
    'django.contrib.sessions.middleware.SessionMiddleware',
    'django.middleware.common.CommonMiddleware',
    'django.middleware.csrf.CsrfViewMiddleware',
    'django.contrib.auth.middleware.AuthenticationMiddleware',
    'django.contrib.auth.middleware.SessionAuthenticationMiddleware',
    'django.contrib.messages.middleware.MessageMiddleware',
    'django.middleware.clickjacking.XFrameOptionsMiddleware',
    'django.middleware.security.SecurityMiddleware',
)

ROOT_URLCONF = 'blogproject.urls'

TEMPLATES = [
    {
        'BACKEND': 'django.template.backends.django.DjangoTemplates',
        'DIRS': [],
        'APP_DIRS': True,
        'OPTIONS': {
            'context_processors': [
                'django.template.context_processors.debug',
                'django.template.context_processors.request',
                'django.contrib.auth.context_processors.auth',
                'django.contrib.messages.context_processors.messages',
            ],
        },
    },
]

WSGI_APPLICATION = 'blogproject.wsgi.application'


# Database
# https://docs.djangoproject.com/en/1.8/ref/settings/#databases

DATABASES = {
    'default': {
        'ENGINE': 'django.db.backends.postgresql_psycopg2',
        'NAME': 'blogproject',
        'USER': '***',
        'PASSWORD': '***',
        'HOST': 'localhost',
        'PORT': '',
    }
}


# Internationalization
# https://docs.djangoproject.com/en/1.8/topics/i18n/

LANGUAGE_CODE = 'en-us'

TIME_ZONE = 'UTC'

USE_I18N = True

USE_L10N = True

USE_TZ = True


# Update database configuration with $DATABASE_URL.
db_from_env = dj_database_url.config(conn_max_age=500)
DATABASES['default'].update(db_from_env)

# Honor the 'X-Forwarded-Proto' header for request.is_secure()
SECURE_PROXY_SSL_HEADER = ('HTTP_X_FORWARDED_PROTO', 'https')

# Allow all host headers
ALLOWED_HOSTS = ['*']

# Static files (CSS, JavaScript, Images)
# https://docs.djangoproject.com/en/1.8/howto/static-files/

STATIC_ROOT = os.path.join(PROJECT_ROOT, 'staticfiles')
STATIC_URL = '/static/'

# Extra places for collectstatic to find static files.
STATICFILES_DIRS = (
    os.path.join(PROJECT_ROOT, 'static'),
)

# Simplified static file serving.
# https://warehouse.python.org/project/whitenoise/
STATICFILES_STORAGE = 'whitenoise.django.GzipManifestStaticFilesStorage'

这是项目的结构

blog-project -- blog -- migrations
                     -- static
                     -- templates
             -- blogproject
             -- blogprojectenv
             -- custom_user
             -- media
             -- .git

有什么想法吗?

【问题讨论】:

    标签: python django heroku deployment collectstatic


    【解决方案1】:

    我今天刚刚更新到 Django 1.10,遇到了完全相同的问题。 你的静态设置也和我的一样。

    这对我有用,运行以​​下命令:

    1. 在部署期间禁用 collectstatic

      heroku config:set DISABLE_COLLECTSTATIC=1

    2. 部署

      git push heroku master

    3. 运行迁移(django 1.10 至少添加了一个)

      heroku run python manage.py migrate

    4. 使用 bower 运行 collectstatic

      heroku run 'bower install --config.interactive=false;grunt prep;python manage.py collectstatic --noinput'

    5. 为未来的部署启用集体静态

      heroku config:unset DISABLE_COLLECTSTATIC

    6. 自行尝试(可选)

      heroku run python manage.py collectstatic

    未来的部署应该从现在开始正常工作

    【讨论】:

    • 在我的情况下,我需要用python manage.py collectstatic 修复一些错误,所以DISABLE_COLLECTSTATIC 本来是一个创可贴......
    • 不知道为什么会这样,但这解决了我的问题,所以 +1。
    • 当我运行heroku config:set DISABLE_COLLECTSTATIC=1 时,我得到了` › 错误:缺少必需的标志:` ` › -a, --app APP 应用程序运行命令针对` ` › 查看更多帮助 --help`
    • 凉亭到底跟什么有什么关系?
    • 得到错误bash: bower: command not found
    【解决方案2】:

    您已将 STATICFILES_DIRS 配置为期望 static 目录与您的 settings.py 文件位于同一目录中,因此请确保它不在其他位置。

    另外,static 目录中是否有任何文件?如果您不这样做,那么 git 将不会跟踪它,因此尽管它存在于本地,但它不会存在于 git 中。通常的解决方案是在目录中创建一个名为 .keep 的空文件,以确保 git 跟踪它。但是一旦你在这个目录中有一些静态文件,那么它就不再是问题了。

    【讨论】:

    • 这对我有用,因为我的 Django 应用程序的 project 包中实际上有一个空的静态目录,因为我所有的静态文件都在主 app 包内的静态目录中。跨度>
    • 这就像一个魅力!我的static 文件夹是空的,添加一个空的.keep 文件对我有用。
    • 我完全忽略了 /static,所以它根本没有添加它。谢谢,我要花很长时间才能弄清楚这一点。
    • 感谢这对我也有帮助。我在 app 文件夹而不是主项目文件夹中创建了静态文件目录。
    • 谢谢,在静态文件夹中放一些东西后,问题修复了
    【解决方案3】:

    不要使用heroku config:set DISABLE_COLLECTSTATIC=1 在heroku 上禁用collectstatic。这只会隐藏错误,不会让您的应用健康。

    相反,最好了解 collectstatic 命令失败的原因,因为这意味着您的设置有问题。

    步骤 1

    在本地运行这两个命令:

    python manage.py collectstatic
    python manage.py test
    

    您应该会看到一条或多条错误消息。大多数情况下,它是一个缺少的变量(例如:STATIC_ROOT),您必须将其添加到您的项目 settings.py 文件中。

    必须添加test 命令,因为某些collectstatic 相关问题只会在test 中出现,例如this one

    第二步

    在本地修复所有错误消息后,再次推送到 heroku。

    疑难解答

    请记住,您也可以直接在 heroku VM 中运行命令。 如果您无法在本地重现,请在 heroku 中运行 collectstatic 命令并直接在您的生产环境中检查发生了什么:

    python manage.py collectstatic --dry-run --noinput
    

    (显然,heroku 控制台也是如此)

    【讨论】:

      【解决方案4】:

      在本地运行 python manage.py collectstatic 并修复所有错误。在我的情况下,存在阻止该命令成功运行的参考错误。

      【讨论】:

      • 附带说明,在 heroku 上使用 whitenoise 和 debug=False 提供静态文件最终变成了一个集群f***,我在搞砸了 2 天后无法让它工作。我最终用静态文件设置了一个 aws s3+cloudfront 并设置了STATIC_URL = 'https://xxxxxxxxxxxxxx.cloudfront.net',它工作得很好。
      【解决方案5】:
      • 发生此错误是因为您没有静态文件 您项目的根目录。

      • 别担心。解决方案是简单

      • 您只需要两步

      第 1 步:打开你的 settings.py 文件并写入

      import os
      from pathlib import Path
      
      BASE_DIR = Path(__file__).resolve().parent.parent
      
      STATIC_ROOT = BASE_DIR / 'staticfiles'
      

      第 2 步: 在项目根目录的终端中运行以下给定命令。 (如果您正在为 Django 项目使用任何虚拟环境,请进入您的虚拟环境,然后进入项目的根目录,然后在给定的命令下运行。)

      python manage.py collectstatic
      

      恭喜:您的问题已解决。

      现在,您可以COMMITPUSH您的“更改”,以便它反映在您的存储库中,然后您就可以开始了。

      【讨论】:

        【解决方案6】:

        这对我有用:

        第 1 步 - heroku config:set DISABLE_COLLECTSTATIC=1
        第 2 步 - git push heroku master

        【讨论】:

        • 你有一个错字:我认为应该是 heroku config:set DISABLE_COLLECTSTATIC=1
        【解决方案7】:

        我面临同样的问题..

        执行此步骤

        1. heroku 配置:设置DISABLE_COLLECTSTATIC=1
        2. git push heroku master
        3. python manage.py collectstatic
        4. python manage.py test
        5. 如果运行测试后出现任何错误..请检查您的 STATIC_ROOT 像这样是正确的 ==> STATIC_ROOT = os.path.join(BASE_DIR, 'static')
        6. 运行 collectstatic 命令后检查所有静态文件是否 存储在您的根目录的 static 目录中。级别(管理.py 目录级别)...
        7. heroku run python manage.py collectstatic
        8. heroku run python manage.py migrate
        9. heroku config:unset DISABLE_COLLECTSTATIC(供将来使用)。

        【讨论】:

          【解决方案8】:

          如果你使用django-heroku

          也许你忘了把这个设置放在行文本 settings.py 的 底部 可以读取所有配置参数

          import django_heroku
          
          django_heroku.settings(locals())
          

          就像documentation:

          Django-Heroku 的使用

          settings.py,在最底部::

          …
          # Configure Django App for Heroku.
          import django_heroku
          django_heroku.settings(locals())
          

          这将自动为您的应用程序配置 DATABASE_URLALLOWED_HOSTS、WhiteNoise(用于静态资产)、Logging 和 Heroku CI。

          p.s: 对不起我的英语不好

          【讨论】:

            【解决方案9】:

            Heroku 制作了一份文档,其中包含有关如何处理此问题的建议 https://devcenter.heroku.com/articles/django-assets

            添加到 settings.py

            BASE_DIR = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
            STATIC_ROOT = os.path.join(BASE_DIR, 'staticfiles')
            STATIC_URL = '/static/'
            STATICFILES_DIRS = (
                os.path.join(BASE_DIR, 'static'),
            )
            

            在项目的根目录中创建一个名为staticfiles 的目录,在其中放置一个网站图标或其他东西,确保 git 跟踪它。然后 collectstatic 命令应该在 heroku 上完成。

            【讨论】:

              【解决方案10】:

              在尝试再次部署应用后遇到了该问题。在我指定这些命令后问题得到了解决:

              $ heroku config:set SECRET_KEY="*secret_key*"
              $ heroku config:set DEBUG_VALUE="True"
              $ heroku config:set EMAIL_USER="*user-email*"
              $ heroku config:set EMAIL_PASS="*pass*"
              

              settings.py 中的这些变量是使用本地环境变量调用的, heroku 在其环境中没有,因此出现错误。

              【讨论】:

              • 这是一个有用的答案。禁用 COLLECTSTATIC 是一种解决方法,就像以下答案之一中提到的那样。我发现对于我的情况,我没有在 Heroku 上设置 SECRET_KEY 配置变量,一旦我把它设置好,一切正常。
              • 这不会在生产中启用调试,这是一个很大的禁忌吗?它应该设置为 false。
              【解决方案11】:

              在我看来,创建 blogproject/static 文件夹时遇到问题。我看到您的博客应用程序中有一个静态文件夹,但它应该在您的 blogproject 文件夹中上一级。

              尝试在您的 blogproject 文件夹中创建一个 static 文件夹,该错误应该会消失。

              【讨论】:

              • 我已经尝试过这样做,但它似乎不起作用。我也在根目录下创建了一个,只是为了试一试,结果是一样的。
              【解决方案12】:

              今天,heroku-django-template 中的 $ pipenv install django$ pip install -r requirements.txt 并非所有要求都正确。

              最新版本的模板包含一个/static 文件夹和一个humans.txt,所以以前的解决方案可能不是问题

              尝试运行$ pipenv install whitenoise,然后运行$ pip freeze &gt; requirements.txt

              如果可行,我也会推荐$ pip install psycopg2 --ignore-installed$ pip freeze &gt; requirements.txt,否则您同样会遇到迁移问题。

              【讨论】:

                【解决方案13】:

                我在部署我的应用时遇到了同样的问题。我意识到我已经更新了我的 pip 版本,安装了一些插件但忘记创建一个新的 requirements.txt 文件。

                在终端中运行 pip freeze &gt; requirements.txt
                运行python manage.py collectstatic
                现在将代码推送到github并部署到heroku服务器

                如果是这样的话,希望这会有所帮助

                【讨论】:

                  【解决方案14】:

                  将这行代码插入到您的 setting.py 文件中。

                  STATIC_ROOT = os.path.join(BASE_DIR, 'static')

                  【讨论】:

                    【解决方案15】:

                    在我的情况下是一个几乎如上所述的错误,在推送导致错误之后,我设置了SECRET_KEY "heroku config:set SECRET_KEY='*************************'"git push heroku main(再次) , heroku run python manage.py migrate , heroku run python manage.py createsuperuser .. 和一切 , heroku open 它奏效了:)

                    【讨论】:

                      【解决方案16】:

                      在我的情况下删除 STATICFILES_DIRS 有效

                      【讨论】:

                        【解决方案17】:
                        heroku config:set DISABLE_COLLECTSTATIC=1 --app #yourappname
                        

                        只需运行命令

                        【讨论】:

                          【解决方案18】:

                          出现此问题是因为 Heroku 尝试运行 manage.py。 在执行 manage.py 时 我们必须像这样写 python manage.py 'some_command'

                          但是 Heroku 尝试将其作为 python manage.py --noinput

                          所以在这种情况下,我们可以对 manage.py 文件进行更改: 最初它看起来像这样:

                           #!/usr/bin/env python
                           """Django's command-line utility for administrative tasks."""
                           import os
                           import sys
                          
                           def main():
                               """Run administrative tasks."""
                               os.environ.setdefault('DJANGO_SETTINGS_MODULE', 
                          'your_project.settings')
                               try:
                                   from django.core.management import execute_from_command_line
                               except ImportError as exc:
                                   raise ImportError(
                                       "Couldn't import Django. Are you sure it's installed and "
                                       "available on your PYTHONPATH environment variable? Did you "
                                       "forget to activate a virtual environment?"
                                   ) from exc
                          
                               execute_from_command_line(sys.argv) # just put this in try block
                          
                           if __name__ == '__main__':
                               main()
                          

                          所以我们把 main.py 改成:

                           #!/usr/bin/env python
                           """Django's command-line utility for administrative tasks."""
                           import os
                           import sys
                          
                           def main():
                               """Run administrative tasks."""
                               os.environ.setdefault('DJANGO_SETTINGS_MODULE', 
                          'your_project.settings')
                               try:
                                   from django.core.management import execute_from_command_line
                               except ImportError as exc:
                                   raise ImportError(
                                       "Couldn't import Django. Are you sure it's installed and "
                                       "available on your PYTHONPATH environment variable? Did you "
                                       "forget to activate a virtual environment?"
                                   ) from exc
                               try:
                                    execute_from_command_line(sys.argv) # just put this in try block
                               except:
                                    pass
                          
                           if __name__ == '__main__':
                               main()
                          

                          【讨论】:

                            【解决方案19】:

                            如果您使用.env 文件和python-decouple,则必须在Heroku app settings &gt; Config Vars 中定义环境变量。否则collectstatic 将不起作用。

                            【讨论】:

                              猜你喜欢
                              • 2021-03-24
                              • 2020-09-11
                              • 1970-01-01
                              • 2014-09-22
                              • 2019-05-10
                              • 1970-01-01
                              • 2021-05-10
                              • 1970-01-01
                              • 2021-09-22
                              相关资源
                              最近更新 更多