我在使用 docker compose 在生产模式下部署的 Django REST 框架应用程序时遇到问题。问题是当我的端点返回带有 ImageField 的模型时,它会返回错误的路径,特别是缺少 url 端口。静态文件在响应 URL 中包含的端口下工作正常。
例如:
当前值:http://127.0.0.1/media/machines/034793bb-8516-45e3-a50a-4e00e7488617.png
预期:http://127.0.0.1:8000/media/machines/034793bb-8516-45e3-a50a-4e00e7488617.png
Settings.py
BASE_DIR = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
TEMPLATE_DIR = os.path.join(BASE_DIR, 'templates')
STATIC_DIR = os.path.join(BASE_DIR,'static')
MEDIA_ROOT = os.path.join(BASE_DIR, 'media')
MEDIA_URL = '/media/'
STATIC_URL = '/static/'
STATIC_ROOT = '/app/static/'
** 码头工人文件 **
FROM python:3.6
ENV PYTHONUNBUFFERED 1
RUN mkdir /app
WORKDIR /app
ADD . /app/
RUN pip install -r requirements.txt
CMD ["gunicorn", "-c", "config/gunicorn/conf.py", "--bind", ":8000", "--chdir",
"app_api.wsgi:application"]
Docker-compose
version: '3.7'
services:
web-service:
build: .
command: bash -c "pip install -r requirements.txt && python manage.py makemigrations && python
manage.py migrate && python manage.py collectstatic --noinput && python manage.py runserver
0.0.0.0:8000"
container_name: app_backend
volumes:
- static:/app/static
- media:/app/media
- .:/app
depends_on:
- db
restart: on-failure:2
db:
image: postgres
container_name: app_postgres
environment:
- POSTGRES_USER=bd_user
- POSTGRES_PASSWORD=password
- POSTGRES_DB=new_database
ports:
- "5432:5432"
volumes:
- postgres:/var/lib/postgresql/data
restart: on-failure:2
nginx:
image: nginx:latest
ports:
- 8000:80
volumes:
- ./config/nginx/conf.d:/etc/nginx/conf.d
- static:/app/static
- media:/app/media
depends_on:
- web-service
volumes:
media:
static:
postgres:
driver: local
Nginx 配置文件
upstream django_server {
server web-service:8000;
}
server {
listen 80;
server_name localhost;
location /static/ {
alias /app/static/;
}
location /media/ {
alias /app/media/;
}
location / {
proxy_pass http://django_server;
proxy_set_header X-Fowarded-For $proxy_add_x_forwarded_for;
proxy_set_header Host $host;
proxy_redirect off;
}
}
** 独角兽配置 **
name = 'docker_django'
loglevel = 'info'
errorlog = '-'
accesslog = '-'
workers = 2
型
class Machine(models.Model):
name = models.CharField(max_length=100, blank=False, null=False)
description = models.CharField(max_length=250, blank=True, null=True)
image = models.ImageField(upload_to = 'machines', default='default.png')
provider = models.TextField(blank=True, null=True)
model = models.CharField(max_length=250, blank=True, null=True)
data_source = models.ForeignKey(DataSource, related_name='data_source', on_delete=models.CASCADE)
def __str__(self):
return self.name
最简单的选择是在nginx中使用$http_host
而不是$host
$host
因为它不包含端口信息。但是,某些客户端的$http_host
可能是空的。
proxy_set_header Host $host;
本 SO 答案中描述的差异
由于nginx对它暴露在docker主机上的端口8000
一无所知,因此无法对其进行设置。一种选择是从请求中获取Host
http 标头(请求中可能不存在,可能在极少数情况下)。
另一种选择是将nginx作为环境变量公开的端口传递到nginx容器中,并在附加到$host
的配置中使用此变量,这对于nginx可以通过在自定义入口点中运行来完成envsubst
。
它适用于静态文件的原因是 - 静态文件 url 包含在模板(不是 api 响应)中,其中包含相对于当前网站根目录的链接(无论它是什么),并且它们不需要build_absolute_uri()
由 django 运行。