将 Django 应用程序部署到 Heroku 时收集静态错误

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

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

这是整个 setings.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

有什么想法吗?

102493 次浏览

在我看来,它在创建 blogproject/static文件夹时遇到了问题。我看到你有一个静态文件夹在你的博客应用程序,但它应该是在你的博客项目文件夹的一个级别。

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

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

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

我今天刚刚升级到姜戈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. 为将来的部署启用 Collecstatic

    heroku config:unset DISABLE_COLLECTSTATIC

  6. 自己试试(可选)

    heroku run python manage.py collectstatic

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

在本地运行 python manage.py collectstatic并修复任何错误。在我的例子中,有一些引用错误阻止了该命令的成功运行。

今天,在 heroku-django-template 和 $ pip install -r requirements.txt$ pipenv install django中,并非所有的需求都得到了恰当的满足。

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

尝试运行 $ pipenv install whitenoise$ pip freeze > requirements.txt

如果这种方法有效,我还建议使用 $ pip install psycopg2 --ignore-installed$ pip freeze > requirements.txt,否则您也会遇到类似的迁移问题。

这对我很有效:

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

Heroku 制作了一份文件,对如何处理这个 https://devcenter.heroku.com/articles/django-assets提出了建议

添加到设置

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 上完成。

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

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

第一步

在本地运行这两个命令:

python manage.py collectstatic
python manage.py test

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

有必要添加 test命令,因为一些与 collectstatic相关的问题只会在 test中出现,例如 这个

第二步

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

故障排除

请记住,您也可以直接在 heroku VM 中运行命令。 如果不能在本地进行复制,那么在 heroku 中运行 Collecstatic 命令,并直接检查生产环境中发生了什么:

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

Launch heroku VM

(显然 Heroku 控制台也是如此)

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

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

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

我也面临同样的问题。

跟着这一步走

  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 命令后,检查所有静态文件 存储在 静电干扰目录中,用于您的根 dir. level (manage.py 长官,水平) ..。
  7. heroku run python manage.py collectstatic.
  8. heroku run python manage.py migrate
  9. heroku config:unset DISABLE_COLLECTSTATIC(供将来使用)。

将这行代码插入到 seting.py 文件中。

STATIC _ ROOT = os.path.join (BASE _ DIR,‘ STATIC’)

在我的例子中,出现了一个与上面描述的差不多的错误,在执行了导致错误的 push-ing 之后,我设置了 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 它起作用了:)

删除 STATICFILES _ DIRS 在我的案例中起作用

heroku config:set DISABLE_COLLECTSTATIC=1 --app #yourappname

只要运行命令

  • 中没有 < strong > staticfiles ,因此发生此错误 您的项目的根目录。

  • 别担心,解决方案是 很简单

  • 你只需要 两步

步骤1: 打开 setings.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

恭喜你,你的问题解决了。

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

在尝试再次部署应用程序时遇到了这个问题。在我指定了以下命令后,问题得到了解决:

$ 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*"

Py 中的这些变量是用本地环境变量调用的, 所以出现了错误。

发生这个问题是因为 Heroku 尝试运行 manage.py。 在执行 manager. 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()

如果使用 Django-heroku

也许您忘记了将这个设置放在行 text setings.py for 的 底部中,因为它可以读取所有的配置参数

import django_heroku


django_heroku.settings(locals())

就像 文件:

Django-Heroku 的用法

settings.py的最底部:

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

这将为您的应用程序自动配置 DATABASE_URLALLOWED_HOSTS、 White噪声(用于静态资产)、日志和 Heroku CI。

附注: 对不起,我的英语不好

如果您使用 .env文件和 python-decouple,您必须在 Heroku app settings > Config Vars中定义环境变量。否则 collectstatic将无法工作。

在测试了发布在这个帖子上的所有内容之后,以下是对我有效的方法:

  • 保持 Heroku 的环境变量设置为0,因为这并不能真正解决问题,只是掩盖它并且破坏你网站的资源
  • 在设置文件中使用 django _ heroku lib 时,包含参数“ staticfiles = False”,如下所示: django_heroku.settings(locals(), staticfiles=False)
  • Py 上的 STATIC _ ROOT 变量应该设置为 STATIC_ROOT = os.path.join(BASE_DIR, 'staticfiles'),BASE _ DIR 设置为 BASE_DIR = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
  • 完成所有这些操作后,运行一个 python manage.py migrateonHerokuCLI,您应该会看到一条关于下载的资产的消息。

在它实际工作之前,Heroku CLI 上的 Collectstatic 命令 python manage.py collectstatic提供了正确的输出,因此请注意,这个命令可能没有错误,但仍然存在错误。