使用docker将django应用程序部署到heroku时,在何处运行collectstatic?

使用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命令在临时文件系

我正在使用Docker将Django应用程序部署到Heroku。当我将
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上工作!这是一个非常优雅的解决方案。