使用docker将django应用程序部署到heroku时,在何处运行collectstatic?
我正在使用Docker将Django应用程序部署到Heroku。当我将使用docker将django应用程序部署到heroku时,在何处运行collectstatic?,django,docker,heroku,static-files,Django,Docker,Heroku,Static Files,我正在使用Docker将Django应用程序部署到Heroku。当我将RUN manage.py collectstatic--noinput放入Dockerfile时,它失败了,因为没有为环境变量DJANGO\u SECRET\u KEY设置值。我的理解是,这是因为配置变量在构建时不可用 当我将collectstatic作为发布命令运行时,它可以正常工作,并成功地复制静态文件。但是,当我点击应用程序url时,它返回一个500错误,因为找不到静态文件。我认为这是因为release命令在临时文件系
RUN manage.py collectstatic--noinput
放入Dockerfile时,它失败了,因为没有为环境变量DJANGO\u SECRET\u KEY
设置值。我的理解是,这是因为配置变量在构建时不可用
当我将collectstatic作为发布命令运行时,它可以正常工作,并成功地复制静态文件。但是,当我点击应用程序url时,它返回一个500错误,因为找不到静态文件。我认为这是因为release命令在临时文件系统上作为dyno运行,因此找不到复制的文件
这似乎是第二十二条军规。将collectstatic放入Dockerfile失败,因为没有可用的配置变量,但将其作为发布命令放入失败,因为只保存构建阶段的文件更改
怎么办
以下是我在settings.py中的静态设置
MIDDLEWARE = [
'django.middleware.security.SecurityMiddleware',
'whitenoise.middleware.WhiteNoiseMiddleware',
...
]
...
STATIC_ROOT = os.path.join(BASE_DIR, 'staticfiles')
STATIC_URL = '/static/'
STATICFILES_DIRS = (
os.path.join(BASE_DIR, 'static'),
)
STATICFILES_STORAGE = 'backend.storage.WhiteNoiseStaticFilesStorage'
from django.core.management.utils import get_random_secret_key
...
SECRET_KEY = os.getenv('DJANGO_SECRET_KEY', default=get_random_secret_key())
Dockerfile
# Pull base image
FROM python:3.7-slim
# Set environment varibles
ENV PYTHONDONTWRITEBYTECODE 1
ENV PYTHONUNBUFFERED 1
# Set work directory
RUN mkdir /code
WORKDIR /code
# Install dependencies
RUN pip install pipenv
COPY Pipfile Pipfile.lock /code/
RUN pipenv install --system
# Copy project
COPY . /code/
## collect static files
RUN mkdir backend/staticfiles
# This fails because DJANGO_SECRET_KEY can't be empty
RUN python manage.py --noinput
# Pull base image
FROM python:3.7-slim
# Set environment varibles
ENV PYTHONDONTWRITEBYTECODE 1
ENV PYTHONUNBUFFERED 1
# Set work directory
WORKDIR /code/
# Install dependencies
RUN pip install pipenv
COPY Pipfile Pipfile.lock .
RUN pipenv install --system
# Copy project
COPY . .
# Collect static files
RUN python manage.py collectstatic --noinput
# run gunicorn
CMD gunicorn hello_django.wsgi:application --bind 0.0.0.0:$PORT
赫罗库
build:
docker:
web: Dockerfile
run:
web: gunicorn backend.config.wsgi:application --bind 0.0.0.0:$PORT
我不使用heroku,因此无法测试,但在运行应用程序之前,您应该能够运行collect static Dockerfile
# Pull base image
FROM python:3.7-slim
# Set environment varibles
ENV PYTHONDONTWRITEBYTECODE 1
ENV PYTHONUNBUFFERED 1
# Set work directory
RUN mkdir /code
WORKDIR /code
# Install dependencies
RUN pip install pipenv
COPY Pipfile Pipfile.lock /code/
RUN pipenv install --system
# Copy project
COPY . /code/
## collect static files
RUN mkdir backend/staticfiles
# This fails because DJANGO_SECRET_KEY can't be empty
RUN python manage.py --noinput
# Pull base image
FROM python:3.7-slim
# Set environment varibles
ENV PYTHONDONTWRITEBYTECODE 1
ENV PYTHONUNBUFFERED 1
# Set work directory
WORKDIR /code/
# Install dependencies
RUN pip install pipenv
COPY Pipfile Pipfile.lock .
RUN pipenv install --system
# Copy project
COPY . .
# Collect static files
RUN python manage.py collectstatic --noinput
# run gunicorn
CMD gunicorn hello_django.wsgi:application --bind 0.0.0.0:$PORT
您也不能在dockerfile中运行collectstatic
,也不能运行事件应用程序,因为它们可以由heroku.yml
运行
build:
docker:
web: Dockerfile
config:
DJANGO_SETTINGS_MODULE: project.settings
run:
web: gunicorn backend.config.wsgi:application --bind 0.0.0.0:$PORT
release:
image: web
command:
- python manage.py collectstatic --noinput
对于您的工作目录,您也不需要mkdir
。只需在dockerfile的早期设置WORKDIR/code/
,之后事情将基于该目录运行
这里有一篇关于这方面的文章 在得到Heroku的支持后,这确实有点像是第二十二条军规 解决方案是将
collectstatic
放在Dockerfile中,这样它就可以在构建时运行,并且文件可以持久保存
通过使用Django的get\u random\u secret\u key
函数设置默认密钥,我们避免了使用密钥配置变量
运行阶段使用Heroku配置变量中的密钥,因此我们实际上不会每次都更改密钥——默认值仅适用于构建过程collectstatic
不在密钥上建立索引,因此这很好
在settings.py中
MIDDLEWARE = [
'django.middleware.security.SecurityMiddleware',
'whitenoise.middleware.WhiteNoiseMiddleware',
...
]
...
STATIC_ROOT = os.path.join(BASE_DIR, 'staticfiles')
STATIC_URL = '/static/'
STATICFILES_DIRS = (
os.path.join(BASE_DIR, 'static'),
)
STATICFILES_STORAGE = 'backend.storage.WhiteNoiseStaticFilesStorage'
from django.core.management.utils import get_random_secret_key
...
SECRET_KEY = os.getenv('DJANGO_SECRET_KEY', default=get_random_secret_key())
他们说它在构建过程中自动运行;您还可以在dockerfile中设置设置模块
ENV DJANGO\u SETTINGS\u MODULE=project.SETTINGS.build
当您构建Docker容器时,它不会自动运行——该链接仅在您未使用Docker时适用。Dockerfile中的collectstatic会导致构建失败,因为构建时collectstatic命令的DJANGO密钥不可用。使用collectstatic作为一个发布命令,它将成功地构建和发布,但实际上找不到静态文件,我相信这是因为在发布期间创建的文件不会持久存在。那篇文章是个谜;我想知道作者是否从未真正测试过它。@RyanKnight,但您会注意到,我显示了作为环境变量添加的设置模块,它将允许django加载设置并执行命令。是的,尝试过,不起作用。问题是配置变量在Heroku上的构建时不可用:@RyanKnight如何从Heroku.yml
为构建传递配置变量?(我已经更新了我的答案以包括这一点)作为构建过程的一部分运行时的问题是找不到设置文件;我在manage.py
中为它设置了一个默认路径,所以这从来都不是问题。问题是访问通过Heroku配置变量设置的环境变量。使用collectstatic失败的配置变量是django密钥,因为它返回“”,但需要一个值。非常感谢!我浪费了几个小时试图让静态文件在Heroku上工作!这是一个非常优雅的解决方案。